news 2026/4/18 10:00:35

MySQL 中的锁你真的了解吗?从行锁到死锁,一篇彻底讲透!

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
MySQL 中的锁你真的了解吗?从行锁到死锁,一篇彻底讲透!

视频看了几百小时还迷糊?关注我,几分钟让你秒懂!


在 Spring Boot 项目中,你是否遇到过这些问题:

  • 用户下单时,库存扣成负数?
  • 两个线程同时修改同一条记录,结果数据错乱?
  • 程序突然卡住,日志显示“等待锁”?

这些问题的背后,往往都是MySQL 锁机制没搞清楚!

今天我们就用Java + Spring Boot + MySQL(InnoDB),手把手带你搞懂:

  • MySQL 到底有哪些锁?
  • 什么情况下会加锁?
  • 如何避免死锁?
  • 如何用代码正确处理并发?

一、为什么需要锁?—— 并发场景下的数据一致性

📌 场景:电商扣库存

// 伪代码 public void reduceStock(Long productId) { Product product = productMapper.selectById(productId); if (product.getStock() > 0) { product.setStock(product.getStock() - 1); productMapper.updateById(product); // 危险! } }

问题
如果两个用户同时下单,都读到stock=1,然后各自减 1 → 最终stock=-1

✅ 解决方案:加锁,确保“读-判断-写”是原子操作。

而 MySQL 的锁,就是实现这种原子性的底层保障。


二、MySQL 锁的分类(InnoDB 引擎)

InnoDB 支持行级锁,这是它比 MyISAM(只支持表锁)更适合高并发的关键!

🔒 1. 按粒度分

类型说明并发性
表锁(Table Lock)锁整张表(MyISAM 默认)❌ 差
行锁(Row Lock)只锁特定行(InnoDB 默认)✅ 高
间隙锁(Gap Lock)锁索引之间的“间隙”防幻读
临键锁(Next-Key Lock)行锁 + 间隙锁InnoDB 默认

💡 InnoDB 在RC(Read Committed)RR(Repeatable Read)隔离级别下,锁行为不同!


🔒 2. 按类型分

锁类型作用典型场景
共享锁(S Lock)允许多个事务读同一行SELECT ... LOCK IN SHARE MODE(已废弃)
排他锁(X Lock)禁止其他事务读写UPDATE / DELETE / SELECT ... FOR UPDATE
意向锁(Intention Lock)表级锁,表示“想对某行加锁”自动加,无需关心

重点掌握:排他锁(X Lock)和间隙锁!


三、实战:Spring Boot 中如何触发 MySQL 行锁?

场景:安全扣库存

✅ 正确做法:使用SELECT ... FOR UPDATE
@Service @Transactional public class OrderService { @Autowired private ProductMapper productMapper; public boolean createOrder(Long userId, Long productId) { // 1. 加排他锁查询库存 Product product = productMapper.selectForUpdate(productId); if (product == null || product.getStock() <= 0) { throw new RuntimeException("库存不足"); } // 2. 扣库存(此时别人无法修改这行) product.setStock(product.getStock() - 1); productMapper.updateById(product); // 3. 创建订单... return true; } }

对应的 Mapper:

@Mapper public interface ProductMapper extends BaseMapper<Product> { @Select("SELECT * FROM product WHERE id = #{id} FOR UPDATE") Product selectForUpdate(@Param("id") Long id); }

🔑 关键:FOR UPDATE会在查询的行上加排他锁(X Lock),直到事务提交才释放。


四、不同隔离级别下的锁行为(重点!)

MySQL 默认隔离级别是RR(Repeatable Read),但很多公司用RC(Read Committed)

隔离级别SELECT ... FOR UPDATE是否加间隙锁?幻读风险
RC(读已提交)❌ 不加间隙锁,只锁具体行✅ 有幻读
RR(可重复读)✅ 加临键锁(行+间隙)❌ 无幻读

🌰 举例:防止“幻读”

user(id PK, name),当前有 id=1,3,5

执行:

SELECT * FROM user WHERE id > 2 AND id < 6 FOR UPDATE;
  • 在 RR 下:不仅锁住 id=3,5,还锁住 (2,3)、(3,5)、(5,+∞) 的间隙,防止别人插入 id=4!
  • 在 RC 下:只锁 id=3,5,别人仍可插入 id=4 → 出现幻读。

💡 如果你用的是RC + 唯一索引等值查询,InnoDB 会退化为记录锁,不加间隙锁。


五、死锁(Deadlock)是怎么发生的?

📌 经典死锁场景

时间事务 A事务 B
T1UPDATE product SET stock=stock-1 WHERE id=1;
T2UPDATE product SET stock=stock-1 WHERE id=2;
T3UPDATE product SET stock=stock-1 WHERE id=2;(等待 B 释放)
T4UPDATE product SET stock=stock-1 WHERE id=1;(等待 A 释放)

💥 结果:互相等待,死锁!

MySQL 会自动检测死锁,并回滚其中一个事务(通常是 undo log 少的那个)。

🔍 如何查看死锁日志?

SHOW ENGINE INNODB STATUS;

LATEST DETECTED DEADLOCK部分能看到详细信息。


六、如何避免死锁?(开发建议)

✅ 1.按固定顺序访问资源

  • 总是先锁 id 小的,再锁 id 大的
  • 避免交叉加锁

✅ 2.减少事务持有锁的时间

  • 事务尽量短
  • 不要在事务中做 RPC 调用、sleep 等耗时操作

✅ 3.加锁前先排序

// 错误:随机顺序 List<Long> ids = Arrays.asList(3L, 1L, 2L); // 正确:先排序 ids.sort(Long::compareTo); for (Long id : ids) { productMapper.selectForUpdate(id); // 按顺序加锁 }

✅ 4.设置超时时间

# application.yml spring: datasource: hikari: connection-timeout: 3000 # 或在 SQL 中

或在 MySQL 层设置:

SET innodb_lock_wait_timeout = 10; -- 等待锁超时 10 秒

七、反例 vs 正例对比

❌ 反例:无锁扣库存(高并发必出错)

// 危险!没有加锁 Product p = productMapper.selectById(1001); if (p.getStock() > 0) { p.setStock(p.getStock() - 1); productMapper.updateById(p); }

✅ 正例:FOR UPDATE+ 事务

@Transactional public void safeReduceStock(Long productId) { Product p = productMapper.selectForUpdate(productId); if (p.getStock() <= 0) throw new RuntimeException("无库存"); p.setStock(p.getStock() - 1); productMapper.updateById(p); }

八、高级技巧:乐观锁 vs 悲观锁

方案原理适用场景
悲观锁(FOR UPDATE先加锁,再操作写多读少,冲突频繁
乐观锁(version 字段)更新时检查 version读多写少,冲突少

乐观锁示例:

ALTER TABLE product ADD COLUMN version INT DEFAULT 0;
// 更新时带 version 条件 int rows = productMapper.updateByVersion( productId, newStock, oldVersion ); if (rows == 0) { throw new RuntimeException("并发修改,请重试"); }

💡建议

  • 库存、余额等强一致性场景 → 用悲观锁
  • 商品信息、点赞数等 → 用乐观锁

九、总结:MySQL 锁核心要点

问题答案
InnoDB 默认锁粒度?行锁
UPDATE会加什么锁?排他锁(X Lock)
RR 隔离级别会加间隙锁吗?会!(防幻读)
如何避免死锁?固定顺序 + 短事务 + 超时
高并发扣库存用什么锁?SELECT ... FOR UPDATE(悲观锁)

记住:锁是双刃剑——不用会数据错乱,滥用会性能下降甚至死锁!


视频看了几百小时还迷糊?关注我,几分钟让你秒懂!

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/17 16:34:23

基于Simulink平台实现无人驾驶运动控制中的非线性模型预测控制算法

基于simulink平台的非线性模型预测控制算法实现代码&#xff0c;无人驾驶运动控制在无人驾驶领域&#xff0c;运动控制是确保车辆安全、高效行驶的核心环节。非线性模型预测控制&#xff08;NMPC&#xff09;算法因其能够处理复杂的非线性系统和约束条件&#xff0c;在无人驾驶…

作者头像 李华
网站建设 2026/4/18 10:00:09

Cyber Triage 3.16 发布 - 通过 Cyber Triage Enterprise 更快开展调查

Cyber Triage 3.16 发布 - 通过 Cyber Triage Enterprise 更快开展调查 Digital Forensics Specialized For Incident Response 请访问原文链接&#xff1a;https://sysin.org/blog/cybertriage-3/ 查看最新版。原创作品&#xff0c;转载请保留出处。 作者主页&#xff1a;s…

作者头像 李华
网站建设 2026/4/18 3:50:36

原圈科技领衔:2026文旅AI市场分析三大平台揭秘,读懂客户抢先机!

本文聚焦2026年文旅行业AI市场分析&#xff0c;精选三大平台进行对比。其中&#xff0c;原圈科技凭借其在全网数据整合、分钟级报告生成及深度行业场景应用方面的突出表现&#xff0c;被普遍视为企业进行AI市场分析的理想选择。该平台在技术实力和功能全面性上表现优异&#xf…

作者头像 李华
网站建设 2026/4/18 3:48:12

DataCMD 怎么部署?用服务器搭建终端数据可视化工具

如果你日常做运维、管服务器,肯定对下面这些场景非常熟悉: 🖥️ top、htop、df -h、iostat 来回敲 😵 数据是有了,但全是文本,靠自己脑补趋势 📉 CPU/负载突然飙高,只能事后翻日志 🧠 想把常用指标“看成图”,却又不想上复杂监控系统 后来我开始用 DataCMD 这种…

作者头像 李华
网站建设 2026/4/18 5:41:20

二分+滑窗|hash

lc2982二分定窗class Solution { public:int maximumLength(string s) {auto check [&](int mid)->bool {unordered_map<char, int> fre_map;for (int i 0; i < s.length();) {int l i;char c s[i];int fre 0;while (s[i] c) {i;}if (i - l > mid) {f…

作者头像 李华