MySQL的事务隔离级别

前言

没什么前言,直接开始吧!
主要想从下面三个方面说一下事务隔离级别:

  1. 为什么会出现事务隔离级别。
  2. 事务隔离级别的种类。
  3. 如何实现事务隔离级别

为什么会出现事务隔离级别?

事务隔离隔离级别,顾名思义,就是事务的隔离级别,主要服务对象就是事务。
事务在高并发的情况下会暴露很多问题,主要有三种:

  1. 脏读
    一个事务读取到另一个事务未提交的数据,别的行未提交的数据也叫脏数据,所以叫脏读
  2. 不可重复读
    在同一个事务中多次读取同一个数据,出现数据不一致,重复读取出现问题,所以叫不可重复读
    不可重复度可以看做脏读的升级版本,读到了别的事务已提交的数据。
  3. 幻读
    在同一个事务中多次读取一定范围内的数据,出现数据行不一致的情况,像是出现幻觉一样,所以叫幻读
    幻读在不可重复读的基础上再上升了一档,不但当前行数据出现了不一致,查找范围内也出现了不一致。

这里比较难区分的是不可重复读和幻读,记住它们两个针对对象不一致即可。

  • 不可重复读针对的主要是单行内的数据,比如在同一个事务中读取id为1的name为a,过一段时间再读取id为1的name却变为b了,就是不可重复读。
  • 幻读针对的主要是一定范围内的数据,比如在同一个事务中读取id大于1但小于10的数据只有三条数据,过一段时间再读取这个范围却变成四条数据,就是幻读。

MySQL是一款支持并发的数据库软件,肯定要解决上面的问题的啊,所以就衍生出了事务隔离这种东西,针对不同的问题,对事务隔离又分了级,就是事务隔离级别

事务隔离级别的种类

针对上面的三个问题,数据库中存在四种事务隔离级别

  1. 未提交读(READ_UNCOMMITTED)
    最低的事务隔离级别,脏读都不能避免。
  2. 已提交读(READ_COMMITED)
    在这个隔离级别下可以避免脏读,但不能避免不可重复读。
  3. 可重复读(REPEATABLE_READ)
    在这个隔离级别下可以避免不可重复读,但不能避免幻读。
  4. 串行化(SERLALIZABLE)
    最高的隔离级别,可以避免并发产生的问题。
事务隔离级别 避免脏读 避免不可重复读 避免幻读
未提交读 × × ×
已提交读 × ×
可重复读 ×
串行化

(在InnoDB中有点特殊,可重复读级别也能在一定程度上避免幻读)
天下没有免费的午餐,越高的隔离级别在进行操作时开销越大,支持的并发越低,为了针对不同的业务场景,MySQL才会对其进行分级。


各隔离级别特性不需要硬记,从中文命名上基本上都能看出来。


事务隔离级别如何实现

这里说的是InnoDB事务隔离级别的实现,毕竟InnoDB是MySQL中支持事务最优秀的存储引擎


未提交读

这个没什么好说的,基本没采取什么措施。不过支持的并发度最高,在不更新数据的时候用这个隔离级别是非常不错的选择。


已提交读和可重复读

这两个可以放在一块说,因为实现的原理差不多。


有锁经验的可能能想到,直接对查询的数据行使用共享锁不就可以防止脏读和不可重复读的问题了吗?的确,直接对数据行使用行锁能有效的避免脏读和不可重复读的问题,但是这也就限制了其他想要修改这行数据的事务。
可能有些人想问了,你想避免脏读和不可重复读不锁住行怎么行?InnoDB就是想不锁住行,还把问题给解决了。
如何在不锁住行的情况下还能避免脏读呢?这就涉及到两个概念:当前读快照读。这两个可以这样理解:

  • 当前读:就是上面上锁的思想,读取数据时顺便将数据行加锁,使其它事务不能修改数据,自然不会产生脏读和不可重复读的情况。
  • 快照读:和名字一样,就是对数据进行进行一次快照读取数据中的内容。因为不需要加锁,又叫非阻塞读。
    为了能支持更高的并发,已提交读和可重复读的隔离级别select只要没有显式加锁(显式加锁:select * from TABLE lock in share mode),采用的就是快照读。只不过采用的策略不一样.
  • 已提交读每次读取都到特定事务版本的数据,没有提交的数据不会对这儿的数据造成影响,这样避免了脏读的发生。
  • 可重复读对特定事务版本的数据进行快照之后,会将读取到的数据快照保存在了另一个地方,读取读取过的数据就会从这个地方中读取(不是口胡,有点绕)。因为事务提交也不会对另存的地方造成影响,这样就避免了不可重复读的发生。
    (可以理解为对数据做了一个备份,备份与原数据没有太大的关联,但注意的是不是将所有数据都做一个备份,那样需要耗费的空间太大,只是将查询过的数据行进行备份,比如查询id为1的数据行,并不会对id为2的数据行进行备份)

快照的实现原理主要涉及以下几个概念,这里就不展开来讲了。

  1. 数据行中的DB_TRX_ID、DB_ROLL_PTR、DB_ROW_ID
    • DB_TRX_ID:事务id,InnoDB中事务都有一个id,这个id是递增的,和事务开启的时间有关系。也就是说,越早开始的事务这个id越小。
    • DB_ROLL_PTR:回滚指针,记录着事务开始的时候在undo日志的位置。
    • DB_ROW_ID:行号。
  2. undo日志
    当我们对数据做了变更操作的时候,就会产生undo记录,undo记录的集合就是undo日志。
    undo中记录的都是更改之前的数据,当我们需要提取事务开始之前的数据时,就需要到undo日志中去查找。
  3. read view
    read view是事务开启时,当前所有事务的一个集合。
可重复读级别下的幻读的防止

InnoDB使用next-key锁保证防止幻读的产生。


next-key锁可以看成由行锁Gap锁两部分组成。行锁不用说了,主要说下Gap锁。


Gap锁

又称间隙锁,顾名思义,就是针对间隙的锁。使用Gap锁的时候是会对数据周围的间隙锁住。这个间隙与数据有关。比如果数据中有:1、3、5、7、9五个数据,那么就可以对下面几个间隙进行锁定:

(-oo, 1]、(1, 3]、(3, 5]、(5, 7]、(7, 9]、(9, +oo]

比如你查询3这个数据行,那么这个事务未提交前其他事务就不能添加大于1小于等于5的数据(假设这个数据不是主键或唯一索引,主键或唯一索引有些特别,待会说)。添加值为4的数据行就会被阻塞住。


查询主键和唯一索引会不会使用Gap锁还得看情况,主要有两种情况:

  • 如果where条件全部命中,则不会用Gap锁,只会加记录锁。
    全部命中:精确查询时所有记录都存在。
  • 如果where部分命中或者全部不命中,则会加上Gap锁。

这也是为了性能考虑,加锁总是得付出代价的,能不加就不加。

为什么主键或唯一索引where条件全部命中可以不用加Gap锁呢?
如果全部命中,InnoDB就会对数据加上记录锁,这个事务提交前别的事务不能删除这些行。因为是唯一索引,所以也不可能添加一样的数据,所以使用记录锁完全满足条件。

串行化

这个是最高的隔离级别,所有的select采用的都是当前读,都会对数据上行锁。

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

推荐阅读更多精彩内容