0

0

mysql如何避免死锁_mysql设计与优化建议

P粉602998670

P粉602998670

发布时间:2026-01-15 11:13:44

|

613人浏览过

|

来源于php中文网

原创

事务中UPDATE多行易死锁,因InnoDB行锁顺序依赖索引扫描路径;若两事务以不同顺序更新同一主键集(如A先5后2、B先2后5),将形成循环等待。

mysql如何避免死锁_mysql设计与优化建议

为什么事务里 update 多行容易触发死锁

MySQL 的 InnoDB 在执行 UPDATE 时默认加的是行级锁,但锁的获取顺序和索引扫描路径密切相关。如果两个事务以不同顺序更新同一组主键(比如事务 A 先改 id=5 再改 id=2,事务 B 反过来),就可能形成循环等待 —— 这就是典型的死锁根源。

更隐蔽的是:即使 SQL 看起来一样,只要执行计划不同(例如一个走主键索引、另一个走二级索引覆盖扫描),实际加锁的行和顺序也可能不一致。

  • 确保所有批量 UPDATE 按主键升序排列后再执行(应用层排序或用 ORDER BY PRIMARY KEY
  • 避免在事务中混合使用 SELECT ... FOR UPDATEUPDATE,尤其当 SELECT 走的是非唯一索引时,可能锁住间隙(Gap Lock)
  • SHOW ENGINE INNODB STATUS\G 查看最近死锁详情,重点关注 TRANSACTION 块里的 lock_modelock_trx_id

唯一索引 vs 普通索引对死锁的影响

更新条件命中唯一索引(如主键或 UNIQUE 键)时,InnoDB 能精确定位单行并只加记录锁;而命中普通索引时,由于无法保证唯一性,InnoDB 会额外加间隙锁(Gap Lock),锁定索引区间,大幅增加锁冲突概率。

比如表 t 有普通索引 idx_status,执行 UPDATE t SET name='x' WHERE status=1,可能锁住所有 status=1 的行及其之间的空隙 —— 即使另一事务只更新 status=2,若它们物理相邻,也可能被波及。

  • 高频更新字段尽量建 UNIQUE 索引(哪怕业务上不强制唯一,也可配合应用逻辑保证)
  • 避免在 WHERE 条件中使用函数或类型隐式转换(如 WHERE DATE(create_time) = '2024-01-01'),这会让索引失效,退化为全表扫描+全表加锁
  • EXPLAIN FORMAT=JSON 确认执行计划是否真的用了预期索引,并观察 key_locks 字段

如何用 SELECT FOR UPDATE 安全地预占资源

SELECT ... FOR UPDATE 不是“保险丝”,它本身就会加锁,且锁的范围由查询条件和索引决定。常见误区是认为“先查再更”比直接 UPDATE 更安全,实际上如果 SELECT 锁得过宽,反而更容易引发死锁。

Kubit.ai
Kubit.ai

一个AI驱动的产品分析平台,为产品和数据团队构建

下载
SELECT id FROM orders WHERE user_id = 123 AND status = 'pending' ORDER BY id LIMIT 1 FOR UPDATE;

这段语句看似只取一行,但如果 user_id + status 没有联合索引,MySQL 可能先扫全表匹配 user_id,再过滤 status,导致锁住大量无关行。

  • 必须为 SELECT FOR UPDATEWHERE 条件建立覆盖索引,且列顺序要匹配查询谓词(如 (user_id, status, id)
  • 禁止在事务中执行无 WHERE 条件或仅用 LIKE '%xxx'SELECT FOR UPDATE,这等于给整张表上锁
  • 如果只是防止重复插入,优先用 INSERT ... ON DUPLICATE KEY UPDATE,它在唯一键冲突时自动转为更新,无需显式加锁

长事务和 autocommit=0 是死锁温床

事务越长,持有锁的时间就越久,其他事务等待的概率指数上升。尤其当 autocommit=0 且忘记 COMMITROLLBACK,锁会一直挂着,后续任何相关操作都可能被堵死。

线上曾见过一个后台任务开启事务后调用外部 HTTP 接口,接口超时 30 秒,期间所有同用户订单更新全部阻塞 —— 根本不是 SQL 写得不好,而是事务边界失控。

  • 所有应用代码中显式开启事务的地方,必须配对 try/finally 或使用上下文管理器确保 COMMIT / ROLLBACK
  • 数据库连接池配置 wait_timeoutinteractive_timeout(建议 ≤ 300 秒),让空闲长连接自动断开,释放锁
  • 监控 INFORMATION_SCHEMA.INNODB_TRX 表,定期告警 trx_state = 'RUNNING'trx_started 超过 60 秒的事务
死锁无法完全消除,但大多数生产环境死锁都源于事务粒度失控、索引设计失当或执行路径不可控 —— 把锁的范围和生命周期看得比 SQL 性能还重,问题就解决了一大半。

相关专题

更多
数据分析工具有哪些
数据分析工具有哪些

数据分析工具有Excel、SQL、Python、R、Tableau、Power BI、SAS、SPSS和MATLAB等。详细介绍:1、Excel,具有强大的计算和数据处理功能;2、SQL,可以进行数据查询、过滤、排序、聚合等操作;3、Python,拥有丰富的数据分析库;4、R,拥有丰富的统计分析库和图形库;5、Tableau,提供了直观易用的用户界面等等。

677

2023.10.12

SQL中distinct的用法
SQL中distinct的用法

SQL中distinct的语法是“SELECT DISTINCT column1, column2,...,FROM table_name;”。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

320

2023.10.27

SQL中months_between使用方法
SQL中months_between使用方法

在SQL中,MONTHS_BETWEEN 是一个常见的函数,用于计算两个日期之间的月份差。想了解更多SQL的相关内容,可以阅读本专题下面的文章。

346

2024.02.23

SQL出现5120错误解决方法
SQL出现5120错误解决方法

SQL Server错误5120是由于没有足够的权限来访问或操作指定的数据库或文件引起的。想了解更多sql错误的相关内容,可以阅读本专题下面的文章。

1095

2024.03.06

sql procedure语法错误解决方法
sql procedure语法错误解决方法

sql procedure语法错误解决办法:1、仔细检查错误消息;2、检查语法规则;3、检查括号和引号;4、检查变量和参数;5、检查关键字和函数;6、逐步调试;7、参考文档和示例。想了解更多语法错误的相关内容,可以阅读本专题下面的文章。

357

2024.03.06

oracle数据库运行sql方法
oracle数据库运行sql方法

运行sql步骤包括:打开sql plus工具并连接到数据库。在提示符下输入sql语句。按enter键运行该语句。查看结果,错误消息或退出sql plus。想了解更多oracle数据库的相关内容,可以阅读本专题下面的文章。

675

2024.04.07

sql中where的含义
sql中where的含义

sql中where子句用于从表中过滤数据,它基于指定条件选择特定的行。想了解更多where的相关内容,可以阅读本专题下面的文章。

572

2024.04.29

sql中删除表的语句是什么
sql中删除表的语句是什么

sql中用于删除表的语句是drop table。语法为drop table table_name;该语句将永久删除指定表的表和数据。想了解更多sql的相关内容,可以阅读本专题下面的文章。

414

2024.04.29

Java 桌面应用开发(JavaFX 实战)
Java 桌面应用开发(JavaFX 实战)

本专题系统讲解 Java 在桌面应用开发领域的实战应用,重点围绕 JavaFX 框架,涵盖界面布局、控件使用、事件处理、FXML、样式美化(CSS)、多线程与UI响应优化,以及桌面应用的打包与发布。通过完整示例项目,帮助学习者掌握 使用 Java 构建现代化、跨平台桌面应用程序的核心能力。

63

2026.01.14

热门下载

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

精品课程

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

共48课时 | 1.8万人学习

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

共3课时 | 0.3万人学习

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

共1课时 | 793人学习

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

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