浅析SQLite的锁机制和WAL技术

锁机制

SQLite基于锁来实现并发控制。SQLite的锁是粗粒度的,并不拥有PostgreSQL那样细粒度的行锁,这也使得SQLite较为轻量级。当一个连接要写数据库时,所有其它的连接都被锁住,直到写连接结束它的事务。

SQLite的数据库连接有5种状态:

状态对应的锁
未加锁
共享(shared)共享锁 
预留(reserved)预留锁
未决(pending)未决锁
排它(exclusive)排它锁

SQL使用锁逐步提升机制,上面的表格从上到下,对应锁的等级逐步提升,等级越高权限就越大。

未加锁:
未和数据库建立连接、已建立连接但是还没访问数据库、已用BEGIN开始了一个事务但未开始读写数据库,处于这些情形时是未加锁状态。

共享:
连接需要从数据库中读取数据时,需要申请获得一个共享锁,如果获得成功,则进入共享状态。

预留:
连接需要写数据库时,首先申请一个预留锁,一个数据库同时只能有一个预留锁,预留锁可以与共享锁共存。获得预留锁后进入预留状态,这时会先在缓冲区中进行需要的修改、更新操作,操作后的结果依然保存在缓冲区中,未真正写入数据库。

未决:
连接从预留升为排它前,需要先升为未决,这时其它连接就不能获得共享锁了,但已经拥有共享锁的连接仍然可以继续正常读数据库,此时,拥有未决锁的连接等待其它拥有共享锁的连接完成工作并释放其共享锁后,提成到排它锁。

排它:
连接需要提交修改时,需要将预留锁升为排它锁,这时其它连接都无法获得任何锁,直到当前连接的排它状态结束。

一个连接读数据的流程如下:

一个连接写数据的流程如下:

死锁

在使用事务的情况下,SQLite的锁机制存在死锁的可能性。
见下面例子:

执行顺序连接A连接B
1BEGIN; 
2 BEGIN;
3 INSERT INTO foo VALUES (‘x’);
4SELECT * FROM foo; 
5 COMMIT;
6 SQL error: database is locked
7INSERT INTO foo VALUES (‘x’); 
8SQL error: database is locked 

执行顺序3:连接B要执行写操作,获得预留锁
执行顺序4:连接A要执行读操作,获得共享锁
执行顺序5:连接B要提交修改,预留锁升级为未决锁
执行顺序6:连接B想要升级为排它锁,必须先等待连接A释放共享锁
执行顺序7:连接A要执行写操作,需要获得预留锁
执行顺序8:连接A获得预留锁失败,必须先等待连接B释放未决锁
于是连接A和连接B相互等待对方,发生死锁。

可以通过正确的使用事务类型来解决以上死锁问题,这里不再陈述。

WAL技术

SQLite在3.7.0开始引入了WAL技术,全称叫Write Ahead Log(预写日志)。

其原理是:修改并不直接写入到数据库文件中,而是写入到另外一个称为WAL的文件中;如果事务失败,WAL中的记录会被忽略,撤销修改;如果事务成功,它将在随后的某个时间被写回到数据库文件中,提交修改。

WAL使用检查点将修改写回数据库,默认情况下,当WAL文件发现有1000页修改时,将自动调用检查点。这个页数大小可以自行配置。

WAL技术带来以下优点:

  • 读写操作不再互相阻塞,一定程度上解决了SQLite在处理高并发上的性能瓶颈
  • 大多数场景中,带来很大的性能提升
  • 磁盘I/O行为更容易被预测

WAL也有一些缺点,不过在iOS开发中影响不大,除非数据达到GB级时,性能才会降低。

如何启用WAL:
如果是直接使用SQLite,需要设置如下编译指示: 

PRAGMA journal_mode=WAL;

如果是使用Core Data,则已经默认开启了WAL模式。

 

http://liuduo.me/2016/04/02/sqlitelockandwal/

 

SQLite基于底层的锁原语设计了4种锁,分别是:

SHARED_LOCK 这是读操作锁, 多个进程可以同时持有该类型的锁,实现并发读操作.
RESERVED_LOCK 这是写操作锁,在给定时间内,只有一个进程可以持有该类型的锁.但是已经持有SHARED_LOCK读锁的其他进程,仍然可以进行读操作,其他进程也可以获得新的SHARED_LOCK读锁.
PENDING_LOCK 这是写操作锁,其他持有SHARED_LOCK锁的进程可以继续进行读操作,但是该锁将阻止其他进程获取新的SHARED_LOCK锁.
EXCLUSIVE_LOCK 这是写操作锁,在给定时间内,只有一个进程持有EXCLUSIVE_LOCK锁,这将阻止其他进程获取任何类型的锁.也就是说,获得该锁的前提是,其他进程没有持有任何锁.
对于SQLite层面的这4种类型的的锁,有2点需要说明的是:

这4种锁依照严格性是依次递增的, 锁之间的状态迁移是有一定顺序的,也是按照严格性进行依次状态迁移的.存在如下几种状态迁移的路径:
---------------------
作者:azurelaker
来源:CSDN
原文:https://blog.csdn.net/azurelaker/article/details/82708573
版权声明:本文为博主原创文章,转载请附上博文链接!

 

转载于:https://www.cnblogs.com/feng9exe/p/10677636.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
SQLite3的锁机制主要分为共享锁(Shared Lock)、排他锁(Exclusive Lock)和共享-排他锁(Shared-Exclusive Lock)三种。 共享锁:当一个事务对某个表或某个表的某个部分加上共享锁时,其他事务也可以对该表或该部分加上共享锁,但是不能对该表或该部分加上排他锁。当一个事务对某个表或某个表的某个部分加上共享锁时,该事务可以读取该表或该部分的数据,但是不能修改该表或该部分的数据。 排他锁:当一个事务对某个表或某个表的某个部分加上排他锁时,其他事务不能对该表或该部分加上任何类型的锁。当一个事务对某个表或某个表的某个部分加上排他锁时,该事务既可以读取该表或该部分的数据,也可以修改该表或该部分的数据。 共享-排他锁:当一个事务对某个表或某个表的某个部分加上共享-排他锁时,如果其他事务已经对该表或该部分加上共享锁,则该事务只能等待其他事务释放该共享锁;如果其他事务已经对该表或该部分加上排他锁,则该事务只能等待其他事务释放该排他锁。当一个事务对某个表或某个表的某个部分加上共享-排他锁时,该事务既可以读取该表或该部分的数据,也可以修改该表或该部分的数据。 SQLite3的锁机制是基于多版本并发控制(MVCC)实现的,即每个事务都可以读取该表的旧版本,而不会被其他事务的修改所影响,同时也可以对该表进行修改,修改后的结果会被保存为新版本,其他事务可以读取到该表的新版本或旧版本。在MVCC机制下,SQLite3可以实现高并发和高效的读写操作。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值