当前位置: 首页 > article >正文

[Mysql]锁总结

Mysql锁总结

  • 锁介绍
    • 锁类型
      • 共享锁(S)
      • 互斥锁(X)
    • 锁粒度
      • 表级锁
        • 表锁
        • 意向锁(Intention Lock)
      • 行级锁
        • 记录锁(Record lock)
        • 间隙锁(Gap lock)
        • Next-Key 锁
        • 插入意向锁(Insert Intention Locks)
    • 其他
      • AUTO-INC Locks
      • Predicate Locks for Spatial Indexes
  • Sql语句加锁介绍
    • SELECT ... FROM
    • SELECT ... FOR UPDATE/ SELECT ... LOCK IN SHARE MODE
    • UPDATE ... WHERE
    • DELETE FROM ... WHERE ...
    • INSERT
    • INSERT ... ON DUPLICATE KEY UPDATE
    • REPLACE
    • INSERT INTO T SELECT ... FROM S WHERE ...
    • CREATE TABLE ... SELECT ...
    • AUTO_INCREMENT
    • FOREIGN KEY
    • LOCK TABLES
  • 死锁场景分析
    • INSERT导致的死锁:回滚导致
    • INSERT导致的死锁:删除导致
    • INSERT ... ON DUPLICATE KEY导致的死锁
    • UPDATE导致的索引:访问资源顺序不同
  • 如何避免死锁

以下介绍主要基于InnoDB存储引擎。

锁介绍

Mysql InnoDB支持多种类型的锁,从不同的维度来看,有不同的锁分类。

锁类型

锁类型来分,分为 **共享锁(S) **和 互斥锁(X)

共享锁(S)

共享锁允许持有该锁的事务去读取某行

互斥锁(X)

互斥锁允许持有该锁的事务去更新/删除某行

不同锁类型之间的兼容关系如下:

S X
S 兼容 冲突
X 冲突 冲突
即 多个事务可以同时对某行加S锁,但不能同时加S-X锁,或者X-X锁。

锁粒度

锁粒度来分,分为 表级锁行级锁:

表级锁

表级锁是指表级别的锁,是加在整张表上。

表锁

表锁即对整张表进行加锁,可以通过lock table来加表锁。
lock table xxx read 对表加S锁
lock table xxx write 对表加X锁

意向锁(Intention Lock)

意向锁也是表级锁,分为意向共享锁(IS)意向互斥锁(IX),表示该事务稍后会在某行记录上请求相应类型的锁。

  • 意向共享锁(IS):表示事务打算在表中某些行上设置共享锁
  • 意向互斥锁(IX):表示事务打算在表中某些行上设置互斥锁

不同锁之间的兼容关系如下:

S IS X IX
S 兼容 兼容 冲突 冲突
IS 兼容 兼容 冲突 兼容
X 冲突 冲突 冲突 冲突
IX 冲突 兼容 冲突 兼容

为什么会有意向锁?InnoDB支持多粒度锁,允许行锁和表锁共存。设想一个场景:当需要对表加X锁时,如果没有意向锁,那么就需要遍历表中的所有记录,判断是否有行锁。

这种方案性能比较低下,如果能在加行锁的时候额外打个标识,这样在需要加表锁时,只需要检查这个标即可,简单方便。这个标就对应这里的 意向锁。

行级锁

行级锁是指记录级别的锁,是加在记录上或记录间。在加行级锁时,默认会加上相应的意向锁。切记行级锁都是加在索引上的,不是加在记录上的。

记录锁(Record lock)

记录锁总是加在索引记录上的。如果表没有索引,会创建默认的聚簇索引,并使用该索引加锁。

间隙锁(Gap lock)

间隙锁是加在两个索引记录之间,或者 第一条索引记录之前,或者最后一条索引记录之后。假设表中索引数据为(10,20,33),则对应的间隙区间为:(-∞,10)、(10,20)、(20,33)、(33,+∞)。

间隙锁是可以并存的,即一个事务获取间隙锁,并不影响其他事务在相同的间隙上获取间隙锁。共享间隙锁和独占间隙锁没有区别,彼此之间不冲突。

间隙锁主要是用来解决幻读的,防止其他事务往“间隙”里面插入数据。间隙锁主要存在于隔离级别为RR的情况,但在RC级别下,对于外键和唯一索引依然会存在间隙锁。这主要是为了保证约束成立。

为什么间隙锁可以共存? 如果从索引中删除某个记录,必须要合并不同事务在该记录上获取的间隙锁。如果不合并则会造成 一个事务清除记录后,另一事务因间隙锁的缘故仍然会认为记录存在。

Next-Key 锁

Next-Key 锁是 间隙锁和记录锁的组合,即锁住记录本身及其之前的间隙,即左开右闭。假设表中索引数据为(10,20,33),则对应的Next-Key间隔区间为:(-∞,10]、(10,20]、(20,33]、(33,+∞)。

对于最后一个间隔,Next-Key锁会锁定索引中最大值之后的间隙以及 supermum伪记录,该伪记录并不是真正的索引记录,但该记录的值高于索引中的任何值。因此,对最后一个间隔的效果就是锁定最大索引之后的间隙。

Mysql的默认隔离级别是RR。在此情形下,在索引查找过程中,默认加上Next-Key锁,但对唯一索引只会加上记录锁。

插入意向锁(Insert Intention Locks)

插入意向锁也是一种间隙锁,在插入记录之前,需要先获取插入意向锁,然后在插入的记录上加X锁。

插入意向锁跟上面的间隙锁是互斥的,但插入意向锁之间是兼容的,即多个不同事务可以在相同的间隙上加插入意向锁。

插入意向锁主要是为了提供并发插入性能,当多个事务同时插入不同记


http://www.kler.cn/news/328780.html

相关文章:

  • C++中,如何使你设计的迭代器被标准算法库所支持。
  • k8s的控制节点不能访问node节点容器的ip地址
  • Scrapy入门
  • 深度学习 Transformer 的标签平滑(Label Smoothing)
  • 计算机视觉小目标检测模型
  • 【Golang】深入解读Go语言中的错误(error)与异常(panic)
  • Base64编码避坑指南
  • Skip、Compose、Flutter和RN
  • 面试金典题3.2
  • 在C语言中,符号有两个主要用途:
  • Rainbond 助力城建智控,从传统开发到敏捷开发转型
  • 算法必学之LRU
  • Gson将对象转换为JSON(学习笔记)
  • 【C++高阶】深入理解C++智能指针:掌握RAII与内存安全的利器
  • 南沙C++信奥赛陈老师解一本通题 2005:【20CSPJ普及组】直播获奖
  • Vue3.X + SpringBoot小程序 | AI大模型项目 | 饮食陪伴官
  • Python知识点:如何使用AWS Greengrass与Python进行边缘计算
  • 64 注意力机制_by《李沐:动手学深度学习v2》pytorch版
  • 【计网】从零开始学习http协议 --- http的请求与应答
  • Stable Diffusion绘画 | 来训练属于自己的模型:素材准备篇
  • 【AI知识点】嵌入向量(Embedding Vector)
  • 明达技术工业级边缘计算网关:智能制造的智慧纽带
  • Docker安装consul + go使用consul + consul知识
  • WaterCloud:一套基于.NET 8.0 + LayUI的快速开发框架,完全开源免费!
  • 墙绘艺术在线市场:SpringBoot实现指南
  • 基于微信小程序爱心领养小程序设计与实现(源码+参考文档+定制开发)
  • 【如何实现一个神经网络】(一)神经元和神经网络
  • C0004.Qt中QComboBox设置下拉列表样式后,下拉列表样式无效的解决办法
  • 【分布式微服务云原生】探索Dubbo:接口定义语言的多样性与选择
  • E35.【C语言】判断大/小端序