0

0

mysql中InnoDB锁的实现与内部机制

P粉602998670

P粉602998670

发布时间:2026-01-27 08:14:03

|

907人浏览过

|

来源于php中文网

原创

InnoDB行锁锁的是索引记录而非数据行:无索引时退化为表锁;主键查询锁聚簇索引记录,二级索引查询可能锁二级索引及对应聚簇索引记录,并在RR级别触发gap lock或next-key lock防幻读。

mysql中innodb锁的实现与内部机制

InnoDB 的行锁到底锁的是“记录”还是“索引”

InnoDB 行级锁不是直接锁数据行,而是锁 索引记录(index record)。这意味着:没有索引的表(即堆表)无法使用行锁,只能退化为表锁;即使有索引,如果 WHERE 条件未命中索引(如全表扫描),也会锁住所有扫描过的索引项——包括间隙(gap)甚至整个范围。

常见错误现象:UPDATE t SET x=1 WHERE id=100; 却导致其他事务更新 id=101 被阻塞——大概率是因为 id 列没建索引,或该查询走的是二级索引+回表路径,实际锁住了二级索引中的记录及对应聚簇索引记录。

  • 主键查询(WHERE id = ?)→ 锁聚簇索引上的单条记录(record lock)
  • 唯一二级索引等值查询 → 先锁二级索引记录,再锁对应聚簇索引记录
  • 非唯一二级索引查询 → 可能触发 next-key lock(记录锁 + 间隙锁),范围更大
  • SELECT ... FOR UPDATEUPDATE/DELETE 在可重复读(RR)下默认用 next-key lock 防幻读

间隙锁(Gap Lock)和临键锁(Next-Key Lock)怎么触发

间隙锁只在事务隔离级别为 REPEATABLE READ 时启用,且仅对已存在的索引间隙加锁,不锁记录本身。它防止其他事务在间隙中插入新记录,是 InnoDB 实现“可重复读”不出现幻读的关键机制。

典型误判场景:执行 SELECT * FROM t WHERE a > 10 AND a 后,另一个事务插入 a = 15 会被阻塞,哪怕表里原本没有 a = 15 的记录——这就是间隙锁生效了。

  • 间隙锁锁定的是索引值之间的“空隙”,例如索引有 [1,5,9],则间隙为 (-∞,1)、(1,5)、(5,9)、(9,+∞)
  • next-key lock = 间隙锁 + 记录锁,覆盖“左开右闭”区间,如 (5,9],用于范围条件
  • 唯一索引的等值查询(含主键)不会加间隙锁,只加 record lock;但范围查询(>, , BETWEEN)会
  • 显式关闭间隙锁:设置 innodb_locks_unsafe_for_binlog = ON(已弃用),或改用 READ COMMITTED 隔离级别

锁升级不存在,但锁数量爆炸很常见

InnoDB 不支持锁升级(lock escalation),不会把大量行锁合并成一个表锁。但它可能因查询条件不精确,导致一次性锁住成百上千个索引项——尤其是 ORDER BY ... LIMIT 配合 FOR UPDATE 时,优化器可能无法精确估算扫描范围,从而锁住远超预期的记录。

Modoer多功能点评系统2.5 精华版 Build 20110710 UTF8
Modoer多功能点评系统2.5 精华版 Build 20110710 UTF8

Modoer 是一款以本地分享,多功能的点评网站管理系统。采用 PHP+MYSQL 开发设计,开放全部源代码。因具有非凡的访问速度和卓越的负载能力而深受国内外朋友的喜爱,不局限于商铺类点评,真正实现了多类型的点评,可以让您的网站点评任何事与物,同时增加产品模块,也更好的网站产品在网站上展示。Modoer点评系统 2.5 Build 20110710更新列表1.同步 旗舰版系统框架2.增加 限制图片

下载

一个真实案例:SELECT * FROM order WHERE status = 'pending' ORDER BY created_at LIMIT 1 FOR UPDATE;,若 status 无索引,就会扫描全表并给每条匹配记录加锁;即使有索引,若 created_at 未包含在联合索引中,也可能导致排序前先锁大量行。

  • 避免锁扩散:确保 WHEREORDER BY 字段被同一联合索引覆盖(如 (status, created_at)
  • 检查执行计划:EXPLAINkeyrows 字段能反映实际扫描与加锁范围
  • SELECT ... LOCK IN SHARE MODE 替代 FOR UPDATE,若业务允许共享访问
  • 监控锁等待:SELECT * FROM information_schema.INNODB_TRX; + INNODB_LOCK_WAITS + INNODB_LOCKS(8.0+ 已移除后者,改用 performance_schema.data_locks

死锁检测不是靠超时,而是主动图遍历

InnoDB 死锁检测是实时的、基于等待图(wait-for graph)的深度优先搜索,一旦发现环就立即回滚其中一个事务(选 undo log 量小的那个),而不是等到 innodb_lock_wait_timeout(默认 50 秒)超时。

所以你看到的 Deadlock found when trying to get lock 错误,是 InnoDB 主动干预的结果,不是等待失败。这也意味着:死锁日志(SHOW ENGINE INNODB STATUS 输出)里一定包含至少两个事务的完整加锁/等待链路。

  • 死锁日志中重点关注 *** (1) WAITING FOR THIS LOCK TO BE GRANTED:*** (2) HOLDS THE LOCK(S): 两段
  • 高频死锁模式:多个事务以不同顺序更新同一组记录(如转账 A→B 和 B→A)
  • 缓解方式:固定 DML 操作顺序(如按主键升序更新)、拆分大事务、减少事务内 SQL 数量
  • 注意:唯一索引冲突插入(Duplicate entry)也可能引发隐式锁和死锁,尤其在高并发 INSERT ... ON DUPLICATE KEY UPDATE 场景

真正难排查的是锁兼容性边缘情况——比如一个事务持有了二级索引的 gap lock,另一个事务尝试在相同间隙插入,却因索引结构分裂导致锁对象变化;这类问题往往需要结合 information_schema.innodb_metrics 中的 lock_row_lock_time_avg 等指标交叉验证。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
mysql修改数据表名
mysql修改数据表名

MySQL修改数据表:1、首先查看数据库中所有的表,代码为:‘SHOW TABLES;’;2、修改表名,代码为:‘ALTER TABLE 旧表名 RENAME [TO] 新表名;’。php中文网还提供MySQL的相关下载、相关课程等内容,供大家免费下载使用。

667

2023.06.20

MySQL创建存储过程
MySQL创建存储过程

存储程序可以分为存储过程和函数,MySQL中创建存储过程和函数使用的语句分别为CREATE PROCEDURE和CREATE FUNCTION。使用CALL语句调用存储过程智能用输出变量返回值。函数可以从语句外调用(通过引用函数名),也能返回标量值。存储过程也可以调用其他存储过程。php中文网还提供MySQL创建存储过程的相关下载、相关课程等内容,供大家免费下载使用。

247

2023.06.21

mongodb和mysql的区别
mongodb和mysql的区别

mongodb和mysql的区别:1、数据模型;2、查询语言;3、扩展性和性能;4、可靠性。本专题为大家提供mongodb和mysql的区别的相关的文章、下载、课程内容,供大家免费下载体验。

281

2023.07.18

mysql密码忘了怎么查看
mysql密码忘了怎么查看

MySQL是一个关系型数据库管理系统,由瑞典MySQL AB 公司开发,属于 Oracle 旗下产品。MySQL 是最流行的关系型数据库管理系统之一,在 WEB 应用方面,MySQL是最好的 RDBMS 应用软件之一。那么mysql密码忘了怎么办呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

515

2023.07.19

mysql创建数据库
mysql创建数据库

MySQL是一个关系型数据库管理系统,由瑞典MySQL AB 公司开发,属于 Oracle 旗下产品。MySQL 是最流行的关系型数据库管理系统之一,在 WEB 应用方面,MySQL是最好的 RDBMS 应用软件之一。那么mysql怎么创建数据库呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

256

2023.07.25

mysql默认事务隔离级别
mysql默认事务隔离级别

MySQL是一种广泛使用的关系型数据库管理系统,它支持事务处理。事务是一组数据库操作,它们作为一个逻辑单元被一起执行。为了保证事务的一致性和隔离性,MySQL提供了不同的事务隔离级别。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

386

2023.08.08

sqlserver和mysql区别
sqlserver和mysql区别

SQL Server和MySQL是两种广泛使用的关系型数据库管理系统。它们具有相似的功能和用途,但在某些方面存在一些显著的区别。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

532

2023.08.11

mysql忘记密码
mysql忘记密码

MySQL是一种关系型数据库管理系统,关系数据库将数据保存在不同的表中,而不是将所有数据放在一个大仓库内,这样就增加了速度并提高了灵活性。那么忘记mysql密码我们该怎么解决呢?php中文网给大家带来了相关的教程以及其他关于mysql的文章,欢迎大家前来学习阅读。

601

2023.08.14

Python 自然语言处理(NLP)基础与实战
Python 自然语言处理(NLP)基础与实战

本专题系统讲解 Python 在自然语言处理(NLP)领域的基础方法与实战应用,涵盖文本预处理(分词、去停用词)、词性标注、命名实体识别、关键词提取、情感分析,以及常用 NLP 库(NLTK、spaCy)的核心用法。通过真实文本案例,帮助学习者掌握 使用 Python 进行文本分析与语言数据处理的完整流程,适用于内容分析、舆情监测与智能文本应用场景。

2

2026.01.27

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
MySQL 教程
MySQL 教程

共48课时 | 1.9万人学习

MySQL 初学入门(mosh老师)
MySQL 初学入门(mosh老师)

共3课时 | 0.3万人学习

简单聊聊mysql8与网络通信
简单聊聊mysql8与网络通信

共1课时 | 811人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号