2016-03-04 106 views
0

我有多重列表,我希望以多線程方式存儲在我的sqlite數據庫使用activeAndroid。我爲每個列表運行一個線程以保持。ActiveAndroid.beginTransaction()和多線程

每個線程的主體看起來像這樣。

ActiveAndroid.beginTransaction(); 
try { 
    for (MyObjToPersist e : myListOfObjToPersist){ 
     e.save(); 
    } 
    ActiveAndroid.setTransactionSuccessful(); 
} 
finally { 
    ActiveAndroid.endTransaction(); 
} 

由於每個線程一個接一個地運行,事務似乎向數據庫添加了一個鎖。

沒有使用交易事情按預期工作,但任務相當緩慢(a hundred time slowest)。

這個問題討論here

有人知道如何避免這種行爲? 謝謝

回答

0

TL; DR:只有一個線程可以修改,但都可以讀取數據。

Can multiple applications or multiple instances of the same application access a single database file at the same time?

多個進程可以同時打開同一個數據庫。多個進程可以同時做一個SELECT。但是,只有一個過程可以隨時對數據庫進行更改。

SQLite使用讀寫器鎖來控制對數據庫的訪問。 (在Win95/98/ME下,缺少對讀/寫鎖的支持,而是使用概率模擬。)但要小心:如果數據庫文件保存在NFS文件系統上,則此鎖定機制可能無法正常工作。這是因爲在許多NFS實現中fcntl()文件鎖定被破壞。如果多個進程可能試圖同時訪問文件,則應避免將NFS數據庫文件放在NFS上。在Windows上,Microsoft的文檔說如果您沒有運行Share.exe守護進程,鎖定可能無法在FAT文件系統下運行。對Windows有很多經驗的人告訴我,網絡文件的文件鎖定非常麻煩並且不可靠。如果他們說的是真的,在兩臺或多臺Windows機器之間共享SQLite數據庫可能會導致意想不到的問題。

我們知道沒有其他嵌入式SQL數據庫引擎支持與SQLite同樣多的併發性。 SQLite允許多個進程一次打開數據庫文件,並允許多個進程一次讀取數據庫。當任何進程想要寫入時,它必須在更新期間鎖定整個數據庫文件。但通常只需要幾毫秒。其他流程只是等待作者完成,然後繼續他們的業務。其他嵌入式SQL數據庫引擎通常只允許一個進程同時連接到數據庫。但是,客戶端/服務器數據庫引擎(如PostgreSQL,MySQL或Oracle)通常支持更高級別的併發,並允許多個進程同時寫入同一數據庫。這在客戶端/服務器數據庫中是可行的,因爲總是有一個可以協調訪問的良好控制的服務器進程。如果您的應用程序需要大量的併發性,那麼您應該考慮使用客戶端/服務器數據庫。但是經驗表明,大多數應用程序所需要的併發性要比設計人員想象的少得多。 當SQLite試圖訪問被另一個進程鎖定的文件時,默認行爲是返回SQLITE_BUSY。您可以使用sqlite3_busy_handler()或sqlite3_busy_timeout()API函數從C代碼調整此行爲。

https://www.sqlite.org/faq.html#q5

+0

好知道。感謝您突出顯示此問題 – aiqency