针对主键索引的 for update 操作有什么用

针对主键索引的 for update 操作有什么用

针对主键索引使用 FOR UPDATE 操作,是 MySQL 高并发编程中保证数据一致性和提升系统性能的核心手段。它的主要作用可以总结为以下三点:

1. 施加排他锁,保证数据强一致性
SELECT ... FOR UPDATE 会在事务中为查询到的行加上排他锁(X锁) 。这意味着在当前事务提交或回滚之前,其他任何事务都无法修改这些行,也无法对这些行再次加锁(比如执行其他的 FOR UPDATE)。这能有效防止在复杂的业务逻辑(如转账、扣库存)中,因并发操作导致的数据被意外篡改或覆盖。

2. 精准的行级锁定,最大化并发性能

这是针对主键索引 最核心的优势。InnoDB 的行级锁是加在索引上的。当你明确使用主键(或唯一索引)进行等值查询时,MySQL 能够精准定位到具体的某一行,并只对该行加锁(即行锁)。

  • 好处:锁的粒度极小,不会波及表中的其他数据。其他事务依然可以自由地读取或修改表里的其他行,系统的并发处理能力极高。
  • 对比 :如果 FOR UPDATE 的查询条件没有走索引(例如对普通字段全表扫描),InnoDB 找不到具体的索引记录,就会退化为锁定整张表(表锁),导致所有其他操作都被阻塞,严重拖垮系统性能。

3. 优化锁的算法,减少不必要的锁定范围

在 MySQL 默认的"可重复读(RR)"隔离级别下,普通的范围查询往往会触发"间隙锁(Gap Lock)"或"临键锁(Next-Key Lock)",把查询条件前后的"间隙"也锁住,以防止幻读。

但是,针对主键的等值查询(例如 WHERE id = 100)是一个特例 。因为主键具有唯一性约束,InnoDB 能够明确知道只会命中这一条记录,因此它会进行锁算法优化,只加最基础的记录锁(Record Lock),而不会加间隙锁。这进一步缩小了锁定的范围,避免了阻塞其他事务在该主键值附近插入新数据。


📌 经典实战场景:银行转账

假设你需要从账户 A 转账 100 元给账户 B,为了防止在扣款和入账的间隙,账户 A 的钱被其他线程同时扣走(导致透支),通常会这样写:

sql 复制代码
START TRANSACTION;

-- 锁定账户A,确保余额在计算期间不会被其他事务修改
SELECT balance FROM account WHERE user_id = 1001 FOR UPDATE;
-- (业务代码检查余额是否充足...)
UPDATE account SET balance = balance - 100 WHERE user_id = 1001;

-- 锁定账户B,进行入账
UPDATE account SET balance = balance + 100 WHERE user_id = 1002;

COMMIT;

在这个例子中,如果 user_id 是主键,那么这两个账户的行会被精准锁定,而不会影响其他用户的正常转账操作。


锁用不好容易死锁,需要我讲讲怎么避免和排查死锁吗?

相关推荐
RainCity1 小时前
Java Swing 自定义组件库分享(六)
java·笔记·后端
xuanjiong1 小时前
DDD架构Repository仓储数据流转全链路详解:Domain与其他层的数据流转
java·系统架构
吴声子夜歌2 小时前
Java——类加载机制
java·开发语言·python
Xiacqi12 小时前
Java 中 String、StringBuffer、StringBuilder 的区别
java
Xiacqi12 小时前
Java 常用集合框架手册
java
笨蛋不要掉眼泪2 小时前
Java并发编程:线程的创建和运行
java·开发语言·jvm
九伯都2 小时前
java编写 agent 入门案例
java·开发语言
环流_2 小时前
redis:持久化rdb
java·数据库·redis
xqqxqxxq2 小时前
Java 线程池(一)
java·开发语言