热点账户更新慢的本质是多事务争抢同一行 x 锁导致串行化瓶颈,需通过分片、缓冲记账、锁优化及最终一致性方案解决。

热点 账户更新慢,本质是多个事务争抢同一行记录的排他锁(X 锁),形成串行化瓶颈。不是 SQL 写得不对,而是架构和执行方式没跟上并发规模。
把“单点压力”打散:库存 / 余额分片
别让所有扣款都指向 account(id=1001, balance=5000) 这一行。改成虚拟分片设计:
- 建子表
account_balance_shard(user_id, shard_id, amount),主键为(user_id, shard_id) - 初始化时把 5000 元拆成 10 份,每份 500 元,shard_id 从 0 到 9
- 每次扣款随机选一个 shard_id 执行
UPDATE …… SET amount = amount - 100 - 查总余额用
SELECT SUM(amount) FROM account_balance_shard WHERE user_id = 1001
10 个分片可降低约 90% 的锁冲突,且无需改业务逻辑主干。
用缓冲记账代替实时更新
高频写入不直接碰核心账户,先落轻量流水表:
- 插入缓冲表
buffer_tx(user_id, op_type, amount, tx_id, status),纯 INSERT 无锁,支持 10 万 + TPS - 异步服务每 10 秒聚合:
SELECT user_id, SUM(amount) FROM buffer_tx WHERE status='pending' GROUP BY user_id - 批量更新核心账户:
UPDATE account SET balance = balance + ? WHERE id = ?,一次更新 N 个用户 - 前端展示余额 = 核心余额 + 未合并的缓冲变动(查 buffer_tx 实时聚合)
锁操作必须精准、短时、可控
避免自以为优化实则放大争用:
- 删掉事务里所有非 DB 操作:HTTP 调用、日志打印、复杂计算——这些会拖长锁持有时间
-
SELECT …… FOR UPDATE只在真正要更新前一刻执行,绝不提前加锁 - WHERE 条件必须走索引:字段类型匹配、不隐式转换、不写
WHERE DATE(create_time)=……这类函数 - MySQL 8.0+ 可加
FOR UPDATE WAIT 1,超 1 秒直接失败,防无限挂起
接受短暂不一致,换系统稳定性
对秒杀、红包等场景,强一致性不是刚需,可用 Redis+ 消息队列解耦:
- 用 Redis Lua 脚本原子扣减:
DECRBY balance_key 100,成功才放行 - 失败直接返回,不穿透到 MySQL
- 成功后发 MQ 消息,消费端控制速率(如 300 条 / 秒)异步落库
- 消息体带唯一
tx_id,DB 层用INSERT IGNORE或ON DUPLICATE KEY UPDATE防重
用户看到“已发放”,数据库延迟几十毫秒写入,体验无感,系统稳如磐石。






























