sqlite3的锁及事务类型
sqlite3总共有三种事务类型:BEGIN [DEFERRED /IMMEDIATE / EXCLUSIVE] TRANSCATION,
五种锁,按锁的级别依次是:UNLOCKED /SHARED /RESERVERD /PENDING /EXCLUSIVE。

当执行select即读操作时,需要获取到SHARED锁(共享锁),
当执行insert/update/delete操作(即内存写操作时),
需要进一步获取到RESERVERD锁(保留锁),
当进行commit操作(即磁盘写操作时),需要进一步获取到EXCLUSIVE锁(排它锁)。

对于RESERVERD锁,sqlite3保证同一时间只有一个连接可以获取到保留锁,也就是同一时间只有一个连接可以写数据库(内存),但是其它连接仍然可以获取SHARED锁,也就是其它连接仍然可以进行读操作(这里可以认为写操作只是对磁盘数据的一份内存拷贝进行修改,并不影响读操作)。

对于EXCLUSIVE锁,是比保留锁更为严格的一种锁,在需要把修改写入磁盘即commit时需要在保留锁/未决锁的基础上进一步获取到排他锁,顾名思义,排他锁排斥任何其它类型的锁,即使是SHARED锁也不行,所以,在一个连接进行commit时,其它连接是不能做任何操作的(包括读)。

PENDING锁(即未决锁),则是比较特殊的一种锁,它可以允许已获取到SHARED锁的事务继续进行,但不允许其它连接再获取SHARED锁,当已存在的SHARED锁都被释放后(事务执行完成),持有未决锁的事务就可以获得commit的机会了。sqlite3使用这种锁来防止writer starvation(写饿死)。

死锁的情况
死锁的情况:当两个连接使用begin transaction开始事务时,
第一个连接执行了一次select操作(已经获取到SHARED锁),
第二个连接执行了一次insert操作(已经获取到了RESERVERD锁),
此时第一个连接需要进行一次insert/update/delete(需要获取到RESERVERD锁),
第二个连接则希望执行commit(需要获取到EXCLUSIVE锁),由于第二个连接已经获取到了RESERVERD锁,根据RESERVERD锁同一时间只有一个连接可以获取的特性,第一个连接获取RESERVERD锁的操作必定失败,
而由于第一个连接已经获取到SHARED锁,第二个连接希望进一步获取到EXCLUSIVE锁的操作也必定失败。就导致了事务死锁。

事务类型的使用原则
在用”begin transaction”显式开启一个事务时,默认的事务类型为DEFERRED,锁的状态为UNLOCKED,即不获取任何锁,如果在使用的数据库没有其它的连接,用begin就可以了。如果有多个连接都需要对数据库进行写操作,那就得使用BEGIN IMMEDIATE/EXCLUSIVE开始事务了。
使用事务的好处是:
1.一个事务的所有操作相当于一次原子操作,如果其中某一步失败,可以通过回滚来撤销之前所有的操作,只有当所有操作都成功时,才进行commit,保证了操作的原子特性;
2.对于多次的数据库操作,如果我们希望提高数据查询或更新的速度,可以在开始操作前显式开启一个事务,在执行完所有操作后,再通过一次commit来提交所有的修改或结束事务。

对SQLITE_BUSY的处理
当有多个连接同时对数据库进行写操作时,根据事务类型的使用原则,我们在每个连接中用BEGIN IMMEDIATE开始事务,即多个连接都尝试取得保留锁的情况,根据保留锁同一时间只有一个连接可以获取到的特性,其它连接都将获取失败,即事务开始失败,这种情况下,sqlite3将返回一个SQLITE_BUSY的错误,如果我们不希望操作就此失败而返回,就必须处理SQLITE_BUSY的情况,
sqlite3提供了sqlite3_busy_handler或sqlite3_busy_timeout来处理SQLITE_BUSY,
对于sqlite3_busy_handler,我们可以指定一个busy_handler来处理,并可以指定失败重试的次数。
而sqlite3_busy_timeout则是由sqlite3自动进行sleep并重试,当sleep的累积时间超过指定的超时时间时,最终返回SQLITE_BUSY。
需要注意的是,这两个函数同时只能使用一个,后面的调用会覆盖掉前次调用。从使用上来说,sqlite3_busy_timeout更易用一些,只需要指定一个总的超时时间,然后sqlite自己会决定多久进行重试以及重试的次数,直到达到总的超时时间最终返回SQLITE_BUSY。并且,这两个函数一经调用,对其后的所有数据库操作都有效,非常方便。

参考:
http://www.sqlite.org/lockingv3.html

Sqlite3并发读写注意事项相关推荐

  1. php中并发读写文件冲突的解决方案(文件锁应用示例)

    php中并发读写文件冲突的解决方案(文件锁应用示例) 参考文章: (1)php中并发读写文件冲突的解决方案(文件锁应用示例) (2)https://www.cnblogs.com/wellsoho/p ...

  2. golang中map并发读写问题及解决方法

    这是一个创建于 2017-03-05 06:02:54 的文章,其中的信息可能已经有所发展或是发生改变. 一.map并发读写问题 如果map由多协程同时读和写就会出现 fatal error:conc ...

  3. 【重难点】【Redis 03】缓存雪崩、缓存穿透、缓存击穿、Redis 的内存过期策略、并发读写和双写

    [重难点][Redis 03]缓存雪崩.缓存穿透.缓存击穿.Redis 的内存过期策略.并发读写和双写 文章目录 [重难点][Redis 03]缓存雪崩.缓存穿透.缓存击穿.Redis 的内存过期策略 ...

  4. makdir php 并发报错_PHP并发读写文件如何解决 - microtime

    关于microtime()函数的问题我分别在程序的开头和结尾用microtime()函数取得当前时间,然后相减得到程序运行时间,在5.2版本的php上没问题,但是在5.3.5.4和5.5三个版本的ph ...

  5. mysql myisam 并发_MySQL的myisam解决并发读写解决方法

    MySQL的myisam解决并发读写解决方法MyISAM在读操作占主导的情况下是很高效的.可一旦出现大量的读写并发,同InnoDB相比,MyISAM的效率就会直线下降,而且,MyISAM和InnoDB ...

  6. sqlite3 解决并发读写冲突的问题

    1 #include "stdafx.h" 2 #include "sqlite3.h" 3 #include <iostream> 4 #incl ...

  7. 【高并发】32位多核CPU并发读写long型数据为何会出现诡异问题?看完这篇我懂了!...

    来自:冰河技术 写在前面 大冰:小菜童鞋,前几天讲的知识点复习了吗? 小菜:复习了,大冰哥,我回去关注了你的公众号,收藏和转发了你的文章,看了好几遍呢!! 大冰:好的,一定要好好复习啊,今天,我们来分 ...

  8. Mysql并发读写解决方案分析--MVCC

    数据库并发场景 所有系统的并发场景都是三种,对于数据库而言为: 读-读:不存在任何问题,也不需要并发控制. 读-写:有线程安全问题,可能会造成事务隔离性问题,也就是脏读,不可重复读,幻读. 写-写:有 ...

  9. sqlite多进程并发读写模式下,返回SQLITE_BUSY错误的处理方法

    SQLite作为一款小型的嵌入式数据库,本身没有提供复杂的锁定机制,无法内部管理多路并发下的数据操作同步问题,更谈不上优化,所以涉及到多路并发的情况,需要外部进行读写锁控制,否则SQLite会返回SQ ...

最新文章

  1. 多普勒效应、多普勒效应、网关、路由器
  2. Tomcat安装配置与基础使用
  3. map文件分析 stm32_浅谈STM32的启动过程
  4. 【转】Android 9 Pie 兼容性常见问题及注意事项
  5. Windows API数据类型表
  6. 单片机之串口通信 485多机通信 红外测试
  7. 解决Windows10 14393版本迅雷崩溃问题
  8. Frank Pfenning
  9. 网络故障排查的思路和方法
  10. control c linux命令,linux中Control+C是什么指令?使用什么命令可以给一个进程发出一个这样的指令?...
  11. 如果可以随意 Emoji 组合,你能让两个表情包碰撞出什么花样?
  12. 单例模式,异常和模块相关
  13. 深入分析QQ键盘保护技术
  14. Springboot物资发放管理系统
  15. Unity流水账8:MeshComponents
  16. PW2312A输入55V,输出0.6A降压IC
  17. 在html里面加入flash,如何在HTML页面插入flash代码
  18. c语言微分方程离散化,PID算法的C语言实现二 PID算法的离散化
  19. 浏览器的页面日志采集
  20. 2021天梯赛 L2-040 哲哲打游戏 题解

热门文章

  1. 《环太平洋》(pacific rim)观后感
  2. OSChina 周日乱弹 —— 我们今天不上班!
  3. 逊哥dp专题 总结(普通dp,斜率优化dp,数位dp)
  4. 【小沐学NLP】Python实现聊天机器人(ELIZA)
  5. 10019---mybatis的缓存机制(一级缓存二级缓存和刷新缓存)
  6. Vue项目启动时卡在98% after emitting CopyPlugin
  7. 码住,这4个宝藏资源网站,让你工作娱乐两不误
  8. 测试工程师需掌握的技能
  9. IT运维如何应对SOX挑战?
  10. 大一学生HTML5期末大作业——基于HTML+CSS中国传统节日-清明节 8页