一、為什么sqlite會(huì)返回database locked而別的數(shù)據(jù)庫(kù)不會(huì)
sqlite可以支持多個(gè)進(jìn)程同時(shí)讀取,但不支持同時(shí)寫入,主要原因是因?yàn)樗J(rèn)采用了串行化的事務(wù)隔離,他在寫數(shù)據(jù)到文件的時(shí)候加了一把文件粒度的排他鎖,這個(gè)時(shí)候是不能并發(fā)讀取的和寫入的,這時(shí)如果有請(qǐng)求就會(huì)進(jìn)入等待。
等待超時(shí)后會(huì)拋出database is locked錯(cuò)誤。這個(gè)和嵌入式?jīng)]什么關(guān)系,嵌入式數(shù)據(jù)庫(kù)同樣可以通過(guò)文件鎖、共享內(nèi)存等方式實(shí)現(xiàn)多進(jìn)程訪問(wèn),通過(guò)快照隔離實(shí)現(xiàn)并發(fā)讀寫。
MySQL里面的myisam引擎其實(shí)也是不支持并發(fā)讀寫的,現(xiàn)在用的比較少,innodb存儲(chǔ)引擎則是通過(guò)MVCC支持并發(fā)讀寫的。
sqlite遇到database is locked問(wèn)題的完美解決
1、使用進(jìn)程或線程間的同步機(jī)制以避免同時(shí)操作;如用信號(hào)量,互斥鎖等(pthread_mutex_lock,
pthread_mutex_unlock),如果你的項(xiàng)目工程較大要求較高的話建議用此方法自行封裝函數(shù)處理同步
2、使用sqlite提供的兩個(gè)busy handler函數(shù),但對(duì)于一個(gè)連接來(lái)說(shuō),只能有一個(gè)busy handle,兩個(gè)函數(shù)會(huì)相互影響,設(shè)
置一個(gè)的同時(shí)會(huì)清除另一個(gè),應(yīng)根據(jù)需要來(lái)選擇。
int sqlite3_busy_handler(sqlite3 *, int (*)(void *, int), void *)
不注冊(cè)此函數(shù)時(shí)默認(rèn)回調(diào)函數(shù)為NULL,清除busy handle,申請(qǐng)不到鎖直接返回;
函數(shù)可以定義一個(gè)回調(diào)函數(shù),當(dāng)出現(xiàn)數(shù)據(jù)庫(kù)忙時(shí)sqlite會(huì)調(diào)用該函數(shù)進(jìn)行延時(shí)并返回非0會(huì)重試本次操作,回調(diào)函數(shù)的第二個(gè)
參數(shù)會(huì)被傳遞為此次因BUSY忙事件而調(diào)用該函數(shù)的次數(shù),因此你完全可以自行控制多少次后(也就是延時(shí)多少后)才真正返回
BUSY;
回調(diào)函數(shù)返回非0,數(shù)據(jù)庫(kù)會(huì)重試當(dāng)前操作,返回0則當(dāng)前操作返回SQLITE_BUSY;
int sqlite3_busy_timeout(sqlite3*, int ms);
不注冊(cè)此函數(shù)時(shí)默認(rèn)超時(shí)等待為0,當(dāng)ms<=0時(shí),清除busy handle,申請(qǐng)不到鎖直接返回;
定義一個(gè)毫秒數(shù),當(dāng)未到達(dá)該毫秒數(shù)時(shí),sqlite會(huì)sleep并重試當(dāng)前操作,
如果超過(guò)ms毫秒,仍然申請(qǐng)不到需要的鎖,當(dāng)前操作返回SQLITE_BUSY;
很多人用這個(gè)函數(shù)沒(méi)有成功,其實(shí)只要你仔細(xì)查看sqlite的源碼就會(huì)發(fā)現(xiàn),
這個(gè)函數(shù)實(shí)際上注冊(cè)了一個(gè)默認(rèn)的sqlite3_busy_handler(sqliteDefaultBusyCallback),而這個(gè)回調(diào)函數(shù)在你的編譯
環(huán)境下可能使得第二個(gè)ms參數(shù)必需要大于1000且是他的整數(shù)倍才有意義,由于此默認(rèn)callback函數(shù)延時(shí)較大,建議自己寫回
調(diào)函數(shù)然后用slite3_busy_handler注冊(cè),這樣就可以自己用自己的延時(shí)函數(shù)或方法進(jìn)行處理了。
延伸閱讀:
二、SQL是什么
Structured Query Language
‘SQL’是結(jié)構(gòu)化查詢語(yǔ)言,是一種用來(lái)操作?RDBMS?的數(shù)據(jù)庫(kù)語(yǔ)言,當(dāng)前關(guān)系型數(shù)據(jù)庫(kù)都支持使用SQL語(yǔ)言進(jìn)行操作,也就是說(shuō)可以通過(guò)?SQL?操作 oracle,sql server,mysql,sqlite 等等所有的關(guān)系型的數(shù)據(jù)庫(kù)
SQL語(yǔ)句主要分為:DQL:數(shù)據(jù)查詢語(yǔ)言,用于對(duì)數(shù)據(jù)進(jìn)行查詢,如select**
DML:數(shù)據(jù)操作語(yǔ)言,對(duì)數(shù)據(jù)進(jìn)行增加、修改、刪除,如insert、udpate、delete**
TPL:事務(wù)處理語(yǔ)言,對(duì)事務(wù)進(jìn)行處理,包括begin transaction、commit、rollback
DCL:數(shù)據(jù)控制語(yǔ)言,進(jìn)行授權(quán)與權(quán)限回收,如grant、revoke
DDL:數(shù)據(jù)定義語(yǔ)言,進(jìn)行數(shù)據(jù)庫(kù)、表的管理等,如create、drop
CCL:指針控制語(yǔ)言,通過(guò)控制指針完成表的操作,如declare cursor對(duì)于web程序員來(lái)講,重點(diǎn)是數(shù)據(jù)的crud(增刪改查),必須熟練編寫DQL、DML,能夠編寫DDL完成數(shù)據(jù)庫(kù)、表的操作,其它語(yǔ)言如TPL、DCL、CCL了解即可SQL 是一門特殊的語(yǔ)言,專門用來(lái)操作關(guān)系數(shù)據(jù)庫(kù)不區(qū)分大小寫