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

彻底讲清 MySQL InnoDB 锁机制:从 Record 到 Next-Key 的全景理解

在真实业务中,你遇到的大多数 MySQL 性能问题、死锁问题,几乎都与“锁”有关。但很多工程师对锁的理解停留在碎片层面:
知道“行锁”“间隙锁”“next-key-lock”,但不知道SQL 是如何触发这些锁的、锁到底锁在哪里、为什么会锁这么多

这一篇文章,我会用工程化思维,带你一次性理解InnoDB 锁机制的全景图

看完以后,你将具备:

✔ 能看懂锁的真实作用
✔ 能从 SQL 推断出锁范围
✔ 能解释死锁发生原因
✔ 能在面试中条理清晰地讲出锁机制


一、为什么 InnoDB 的锁机制这么复杂?

MySQL 采用MVCC + 锁实现事务隔离,其中最关键的隔离级别是:

  • RC:读已提交

  • RR:可重复读(默认)

RR 是企业最常见的隔离级别,它要解决“幻读”问题。
于是有了三个锁:

  • 记录锁(Record Lock)

  • 间隙锁(Gap Lock)

  • Next-Key Lock(Record + Gap)

所有复杂问题都来自这个组合。


二、InnoDB 的三种核心锁(理解它们的作用和触发条件)

Record Lock:真实行上的锁

锁的对象:一条真实存在的记录
触发场景:精确命中唯一索引

例如:

SELECT * FROM user WHERE id = 10 FOR UPDATE;

只锁(10]—— 单条记录。

特点:不会锁间隙,因此不会阻止插入。


Gap Lock:只锁“间隙”,不锁数据

作用:阻止“间隙内插入新数据”,防止幻读。

例如:
索引中已有值:

10 --- 20 --- 30

SQL:

SELECT * FROM user WHERE age > 20 FOR UPDATE;

Gap Lock 会锁住:

(20, 30) (30, +∞)

重点:Gap Lock 不锁记录,只锁区间。


Next-Key Lock:Record + Gap 的组合锁

RR 下范围查询的默认锁模式:

(prev_key, record_key]

例如:(假设索引有 10、20、30)

SELECT * FROM t WHERE age BETWEEN 15 AND 25 FOR UPDATE;

锁住的区间:

(10,20] (20,30]

作用:
✔ 锁住命中的记录
✔ 锁住记录前的 gap → 阻止插入

这就是为什么 RR 隔离级别能规避幻读。


三、锁到底由哪些 SQL 触发?

“ SQL → 锁类型” 映射表:

SQL 场景索引情况锁类型原因
WHERE id = ?(唯一键)精确命中Record Lock不需要锁 gap
WHERE id = ?(普通索引)精确匹配,但非唯一Next-Key Lock防止幻读
WHERE age > ?/< ?范围扫描Next-Key Lock必须锁 gap
BETWEEN范围查询范围扫描Next-Key Lock防止插入
无索引过滤全表扫描大量 Record Lock每条记录都会被锁
LIKE '%abc'无法走索引表锁风险全表扫描

一句话总结:

能精确锁住记录 → Record Lock
需要范围扫描 → Next-Key Lock
范围扫描一定会锁 gap → Gap Lock


四、锁具体加在什么区间?

假设索引中有如下值:

10 ---- 20 ---- 30 ---- 40

来看不同 SQL 加的锁👇

WHERE id = 20 FOR UPDATE

锁:

(10, 20]

但如果字段是主键/唯一键,会优化成:

[20]

WHERE id > 20 FOR UPDATE

锁:

(20,30) (30,40) (40,+∞)

WHERE id BETWEEN 15 AND 35 FOR UPDATE

锁:

(10,20] (20,30] (30,40]

无索引条件

SELECT * FROM user WHERE name='xxx' FOR UPDATE;

锁住所有记录:

[10], [20], [30], [40]

→ 大量锁冲突发生的根源。


五、总结

  1. InnoDB 的锁永远基于索引。

  2. 无法精确匹配记录,就会使用 Next-Key Lock。

  3. 范围查询一定会带 gap 锁。

掌握这三点后,死锁、锁等待、幻读问题都能一眼看穿。

http://www.rkmt.cn/news/89218.html

相关文章:

  • MCU的启动流程你了解么?
  • I2C通信最全面的讲解:从协议到硬件设计
  • 【题解】Luogu P10752 [COI 2024] Sirologija
  • Python字符串:别只用来打印!这5个高级用法让代码效率翻倍
  • 【题解】Atcoder ABC432 C
  • 赶due党救急!论文降重2小时搞定,不熬夜
  • 计算机论文模板推荐:8大平台+AI修改工具
  • 期待回家,顺便写点年度总结
  • E No address added out of total 1 resolved地址绑定失败: No address added out of total 1 resolved errors:
  • JavaScript 异常原因(Error Cause):实现分布式系统错误链追踪的序列化与反序列化
  • JavaScript 记录(Records)与 元组(Tuples):实现堆内存中不可变复合数据结构的内存布局
  • 线程并发编程,同步与互斥机制
  • Python列表与元组:搞懂这3个核心差异,再也不纠结用哪个
  • MQ消息队列相关知识与对比
  • 完整教程:PPT导出为图片的格式选择:JPG与PNG的区别
  • 代码随想录算法训练营第三十二天 | 完全背包、518. 零钱兑换 II、377. 组合总和 Ⅳ、卡码网57. 爬楼梯
  • 基于深度学习的文物图像修复系统
  • JavaScript 引擎中的分支预测器(Branch Predictor)友好性:如何写出减少 CPU 误判的代码
  • Day 37 - 早停策略与模型权重的保存
  • 【SOVD】软件定义汽车时代的诊断新范式
  • 最全词典整合收录:打造专业英语学习利器
  • C盘哪些文件可以删除?
  • 18、深入了解 Linux 文件系统:导航与分区指南
  • PLM系统更专业化:更适配汽车电子芯片半导体研发的高标准管理选择——全星研发项目管理APQP软件系统应用解析
  • 磁盘清理工具没反应怎么办
  • 从入门到转行:网络安全自学与跳槽的终极建议
  • PyTorch Geometric中TUDataset加载问题全解析:从诊断到实战
  • 12月12日总结 - 作业----
  • Blade构建系统终极指南:新手快速上手指南
  • Extreme Programming--front-end and back-end separation contacts programming