sqlite3报错:database is locked

爱躺平的咸鱼 2024-10-07 09:37:00 阅读 52

问题描述:

使用visual studio2022创建动态库,动态库中包含向sqlite3数据库插入数据的函数,在测试函数时偶尔出现database is locked的错误提示。

思路:

1、最开始以为是代码写的有问题,可能对于某些资源没有释放,经过检查后发现代码中某些情况下没有释放stmt,修改后重新测试依然没有解决问题。

2、后来根据locked想到了锁,那么是不是被程序中其他线程锁住了呢?很遗憾,我的程序是单线程的,并不是这个问题。

3、经过网上查询资料发现,依然有很大可能是被其他某个东西锁住了。不一定是线程,也许是其他进程,于是发现了我另一个软件(sqlite expert)打开数据库文件后一直在监听数据,时不时会更新数据。那么会不会是因为这个软件一直在监听数据导致的呢?但是很奇怪的是,这个软件只是读数据,而不会去写数据,真的是这个软件导致的问题吗?

问题原因

经过查询sqlite3的锁机制发现:sqlite3并不支持并发写,在某一时刻只能有一个连接去写数据。

sqlite3的锁包含五种状态:未加锁、共享锁、保留锁、未决锁、排他锁。

首先,当所有的连接都没有连接到数据库时,处于未加锁状态;

其次,当某个连接试图去读取数据库数据时,此时处于共享锁状态,此时依然允许其他连接申请共享锁;

接着,当某个连接试图去写数据时,需要申请保留锁,意味着该连接已经准备好写数据了(但还没写),此时依然允许有连接申请共享锁;

最后,试图写数据的连接需要获取写锁,由于sqlite3只允许一个线程写入数据,因此需要将保留锁升级成排他锁,从而写入数据。而当它试图升级为排它锁但还没获得时,则先将锁升级为未决锁,此时将不再允许有新的连接获得共享锁,当所有的已经获得共享锁的连接读取完数据释放共享锁后,此时试图写入数据的连接将未决锁升级成排它锁,从而写入数据。

总结

那么此时回看之前的问题,由于sqlite expert会时不时自动刷新数据,因此,当sqlite expert没有对数据进行刷新时,数据可以正常的插入数据库文件中,但是当sqlite expert刷新数据时,就会导致database is locked,因为只要有一个连接持有共享锁,都无法写入数据。这也是为什么偶尔该错误的原因。后来将sqlite expert换成了sqlite studio,sqlite studio不会自动刷新数据,需要手动点击刷新按钮才会刷新数据,因此用sqlite studio打开数据库文件并没有出现类似的问题(不要在连续插入数据时点击刷新数据的按钮,否则也会出现上述问题)。



声明

本文内容仅代表作者观点,或转载于其他网站,本站不以此文作为商业用途
如有涉及侵权,请联系本站进行删除
转载本站原创文章,请注明来源及作者。