mysql中的各种锁把我搞糊涂啦~

2021-02-21  本文已影响0人  java小杰要加油

大家好,我是公众号:java小杰要加油, 今天来分享一个关于mysql的知识点——mysql中的锁

事务并发访问情况

读-读 情况

写-写 情况

Q: 能描述一下两个事务并发修改同一条数据时,mysql这个锁是怎么避免脏写的吗?

A :事务T1在更改这条数据前,就先内存中生成一把锁与此数据相关联(is_waiting为false,代表没有等待),然后咔咔一顿操作更改数据,这个时候,事务T2来了,发现此记录已经有一把锁与之相关联了(就是T1那一把锁),然后就开始等待(is_waiting为true代表正在等待),事务T1更改完数据提交事务后,就会把此事务对应的所结构释放掉,然后检测一下还有没有与此记录相关联的锁,结果发现T2还在苦苦的等待,就把T2的锁结构的(is_waiting为false,代表没有等待)然后把T2事务对应的线程唤醒,T2获取锁成功继续执行,总体流程如上。

读-写 /写-读 情况

在读-写 / 写 -读的情况下会出现脏读,不可重复读,幻读的现象,不同的隔离级别可以避免不同的问题,具体相关内容可以看小杰的这篇文章 京东面试官问我:“聊聊MySql事务,MVCC?”

不过贴心的我还是列出来了 注:√代表可能发生,×代表不可能发生

隔离级别 脏读 不可重复读 幻读
读未提交(read uncommitted RU)
读提交(read committed RC) ×
可重复读(repeatable read RR) × ×
串行化(serializable ) × × ×

但是 RR在某些程度上避免了幻读的发生

怎么避免脏读、不可重复读、幻读这些现象呢?其实有两种方案

  • mvcc里面最重要的莫过于ReadView了,它的存在保证了事务不可以读取到未提交的事务所作的更改,避免了脏读。
  • 在RC隔离级别下,每次select读操作都会生成ReadView
  • 在RR隔离级别下,只有第一次select读操作才会生成ReadView,之后的select读操作都复用这一个ReadView

某些业务场景不允许读取旧记录的值,每次读取都要读取最新的值。 例如银行取款事务中,先把余额读取出来,再对余额进行操作。当这个事务在读取余额时,不允许其他事务对此余额进行访问读取,直到取款事务结束后才可以访问余额。所以在读数据的时候也要加锁

锁分类

当使用读写都加锁这个方案来避免并发事务写-写读-写写-读时而产生的脏读不可重复读幻读现象时,那么这个锁它就要做到,读读时不相互影响,上面三种情况时要相互阻塞,这时锁也分了好几类,我们继续往下看

锁定读

他们之间兼容关系如下 √代表可以兼容,×代表不可兼容

兼容性 S锁 X锁
S锁 ×
X锁 × ×

事务T1获取某记录的S锁后,

事务T1获取某记录的X锁后,

锁定读语句

<pre class="custom" data-tool="mdnice编辑器" style="margin-top: 10px; margin-bottom: 10px; border-radius: 5px; box-shadow: rgba(0, 0, 0, 0.55) 0px 2px 10px;">`SELECT .. LOCK IN SHARE MODE # 对读取的记录添加S锁

SELECT .. FOR UPDATE # 对读取的记录添加X锁` </pre>

多粒度锁

前面提到的锁都是针对记录的,其实一个事务也可以在表级进行加锁(S锁、X锁)

可是怎么可能平白无故的就给表加锁呢,难道没什么条件吗?答案是肯定有条件的

但是这个怎么确保呢?难道要一行一行的遍历表中的所有数据吗?当然不是啦,聪明的大佬们想出了下面这两把锁

  • 意向共享锁(Intention Shared Lock):简称IS锁,当事务准备在某记录上加S锁时,需要先在表级别加上一个IS锁
  • 意向独占锁(Intention Exclusive Lock):简称IX锁,当事务准备在某记录上加X锁时,需要先在表级别加上一个IX锁

让我们来看下加上这两把锁之后的效果是什么样子的

image image

然后 经过上面的操作之后

image image

这几种锁的兼容性如下表

兼容性 IS锁(表级锁) S锁 IX锁(表级锁) X锁
IS锁(表级锁) ×
S锁 × ×
IX锁(表级锁) × ×
X锁 × × × ×
  • IS、IX锁都是表级锁,他们可以共存。
  • 他们的提出仅仅是为了在之后加表级别的S锁或者X锁时可以快速判断表中的记录是否被上锁,避免用遍历的方式来查看一行一行的去查看而已

InnoDB中的行级锁

Record Lock(记录锁)

Gap Lock(间隙锁)

Next-Key Lock(记录锁+间隙锁)

Insert Intention Lock(插入意向锁锁)

锁内存结构

当然不是!

一个锁结构

如果被加锁的记录符合下面四条状态的话,那么这些记录的锁则会合到一个锁结构

image

锁结构信息

然后我们再来依此看下这个所结构每个部分的信息都是什么意思

image image

举个例子

事务T1 要给user表中的记录加锁,假设这些记录存储在表空间号为20,页号为21的页面上,T1给id=1的记录加S型Record Lock锁,假如当前页面一共有5条记录(3条用户记录和2条伪记录)

过程:先给表加IS锁,不过我们现在不关心,只关心行级锁, 具体生成的所结构如下图所示

image

最后

好文推荐

image
上一篇下一篇

猜你喜欢

热点阅读