SQLite学习(六)

一、数据库锁

SQLite 采用粗放型的锁。当一个连接要写数据库,所有其它的连接被锁住,直到写连接结束了它的事务。 SQLite 有一个加锁表,来帮助不同的写数据库都能够在最后一刻再加锁, 以保证最大的并发性。

SQLite 使用锁逐步上升机制,为了写数据库,连接需要逐级地获得排它锁。
SQLite有5个不同的锁状态:未加锁(UNLOCKED)、共享(SHARED)、保留 (RESERVED)、未决(PENDING)和排它(EXCLUSIVE)。

注意:每个数据库连接在同一时刻只能处于其中一个状态。每种状态(未加锁状态除外)都有一种锁与之对应。

下面具体讲解下各个锁:
1、最初的状态是未加锁状态,在此状态下,连接还没有存取数据库。当连接到了一个数据库, 甚至已经用 BEGIN 开始了一个事务时,连接都还处于未加锁状态。
2、未加锁状态的下一个状态是共享状态。为了能够从数据库中读(不写)数据,连接必须首先进入共享状态,也就是说首先要获得一个共享锁。多个连接可以同时获得并保持共享锁,也就是说多个连接可以同时从同一个数据库中读数据。但哪怕只有一个共享锁还没有释放,也不允许任何连接写数据库。
3、如果一个连接想要写数据库,它必须首先获得一个保留锁。一个数据库上同时只能有一个保留锁。保留锁可以与共享锁共存,保留锁是写数据库的第 1 阶段。保留锁即不阻止其它拥有共享锁的连接继续读数据库,也不阻止其它连接获得新的共享锁。
4、一旦一个连接获得了保留锁,它就可以开始处理数据库修改操作了,尽管这些修改只能在缓冲区中进行,而不是实际地写到磁盘。对读出内容所做的修改保存在内存缓冲区中。 当连接想要提交修改(或事务)时,需要将保留锁提升为排它锁
5、为了得到排它锁,还必须首先将保留锁提升为未决锁。获得未决锁之后,其它连接就不能再获得新的共享锁了,但已经拥有共享锁的连接仍然可以继续正常读数据库。此时,拥有未决锁的连接等待其它拥有共享锁的连接完成工作并释放其共享锁。
6、一旦所有其它共享锁都被释放,拥有未决锁的连接就可以将其锁提升至排它锁,此时就可以自由地对数据库进行修改了。所有以前对缓冲区所做的修改都会被写到数据库文件。

二、死锁

为什么需要了解锁的机制呢?为了避免死锁。
考虑下面表所假设的情况。两个连接——A 和 B——同时但完全独立地工作于同一个数据库。

A连接 B连接
sqlite> BEGIN;
sqlite> BEGIN;
sqlite> INSERTINTO foo VALUES('x');
sqlite> SELECT* FROM foo;
sqlite> COMMIT;
SQL error: database is locked
sqlite> INSERTINTO foo VALUES('x');
SQL error: database is locked

两个连接都在死锁中结束。B 首先尝试写数据库,也就拥有了一个未决锁。A 再试图写,但当其 INSERT 语句试图将共享锁提升为保留锁时失败。
为了讨论的方便,假设连接 A 和 B 都一直等待数据库可写。那么此时,其它的连接甚至都 不能够再读数据库了,因为 B 拥有未决锁(它能阻止其它连接获得共享锁)。那么时此,不仅 A 和 B 不能工作,其它所有进程都不能再操作此数据库了。
如果避免此情况呢?当然不能让 A 和 B 通过谈判解决,因为它们甚至不知道彼此的存在。 答案是采用正确的事务类型来完成工作。

三、事物

事务的种类:

SQLite 有三种不同的事务,使用不同的锁状态。事务可以开始于:DEFERRED、MMEDIATE 或 EXCLUSIVE。事务类型在 BEGIN 命令中指定:

BEGIN [ DEFERRED | IMMEDIATE| EXCLUSIVE ] TRANSACTION;

1、一个 DEFERRED 事务不获取任何锁(直到它需要锁的时候),BEGIN 语句本身也不会做什么 事情——它开始于 UNLOCK 状态。默认情况下 就是这样的,如果仅仅用 BEGIN 开始一个 事务,那么事务就是DEFERRED 的,同时它不会获取任何锁;
2、当对数据库进行第一次读操作时,它会获取 SHARED 锁;同样,当进行第一次写操作时,它会获取 RESERVED锁。 由 BEGIN 开始的IMMEDIATE 事务会 尝试 获取 RESERVED锁 。如果成功, BEGIN IMMEDIATE保证没有别的连接可以写数据库。但是,别的连接可以对数据库进行读操作; 但是,RESERVED锁会阻止其它连接的 BEGIN IMMEDIATE或者 BEGIN EXCLUSIVE 命 令,当其它连接执行上述命令时, 会返回 SQLITE_BUSY 错误。这时你就可以对数据库进 行修改操作了,但是你还不能提交,当你 COMMIT 时,会返回SQLITE_BUSY 错误,这意 味着还有其它的读事务没有完成,得等它们执行完后才能提交事务。
3、EXCLUSIVE 事务会试着获取对数据库的 EXCLUSIVE 锁。这与 IMMEDIATE类似,但是一 旦成功,EXCLUSIVE 事务保证没有其它的连接,所以就可对数据库进行读写操作了。 上节那个例子的问题在于两个连接最终都想写数据库,但是它们都没有放弃各自原来的锁, 最终,SHARED 锁导致了问题的出现。
4、如果两个连接都以 BEGIN IMMEDIATE开始事务, 那么死锁就不会发生。在这种情况下,在同一时刻只能有一个连接进入BEGIN IMMEDIATE, 其它的连接就得等待。BEGIN IMMEDIATE和 BEGIN EXCLUSIVE 通常被写事务使用。就像同步机制一样,它防止了死锁的产生。 基本的准则是:如果你正在使用的数据库没有其它的连接,用 B EGIN 就足够了。但是,如 果你使用的数据库有其它的连接也会对数据库进行写操作,就得使用 BEGINIMMEDIATE 或BEGIN EXCLUSIVE开始你的事务。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,245评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,749评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,960评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,575评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,668评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,670评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,664评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,422评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,864评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,178评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,340评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,015评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,646评论 3 323
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,265评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,494评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,261评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,206评论 2 352

推荐阅读更多精彩内容