news 2026/4/23 17:23:35

MySQL高并发下 SELECT ... FOR UPDATE 性能差的庖丁解牛

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
MySQL高并发下 SELECT ... FOR UPDATE 性能差的庖丁解牛

MySQL 高并发下SELECT ... FOR UPDATE性能差,本质是行锁竞争 + 事务持有时间过长导致的锁等待与吞吐下降。它并非“不好用”,而是在错误场景下被滥用


一、核心原理:FOR UPDATE如何工作?

▶ 1.加锁机制
  • InnoDB 行锁
    • SELECT ... FOR UPDATE会对结果集所有行排他锁(X Lock)
    • 其他事务无法读取(若未开启 MVCC 快照读)或修改这些行
  • 锁范围
    • 若无索引 →全表扫描 + 锁所有行(灾难!)
    • 若有索引 →仅锁命中行
▶ 2.事务生命周期
MySQL事务2事务1MySQL事务2事务1BEGINSELECT * FROM seats WHERE id=100 FOR UPDATE返回数据 + 加锁SELECT * FROM seats WHERE id=100 FOR UPDATE阻塞(等待 T1 释放锁)COMMIT返回数据

💡核心认知
FOR UPDATE的性能 = 锁粒度 × 事务时长 × 并发度


二、性能瓶颈:三大致命问题

▶ 1.锁粒度过大
  • 场景
    -- 无索引字段查询SELECT*FROMordersWHEREuser_id=123FORUPDATE;
  • 后果
    • 全表扫描 → 锁住所有行(即使只返回 1 行)
    • 并发度 ≈ 1(其他事务全部阻塞)
▶ 2.事务持有时间过长
  • 场景
    $pdo->beginTransaction();$seat=$pdo->query("SELECT * FROM seats WHERE id=100 FOR UPDATE")->fetch();// 调用第三方支付 API(耗时 2 秒!)$paymentResult=callPaymentAPI($seat);if($paymentResult){$pdo->exec("UPDATE seats SET status=2 WHERE id=100");}$pdo->commit();
  • 后果
    • 行锁持有 2 秒 → 其他请求排队等待
    • 吞吐量从 1000 QPS 降至 50 QPS
▶ 3.死锁风险
  • 场景
    • 事务 A 锁 seat 100 → 尝试锁 seat 101
    • 事务 B 锁 seat 101 → 尝试锁 seat 100
  • 后果
    • MySQL 检测到死锁 → 回滚其中一个事务 → 重试成本高

三、工程优化:四层解决方案

▶ 方案 1:缩小锁粒度(最有效)
  • 必须为 WHERE 字段加索引
    -- 添加索引ALTERTABLEseatsADDINDEXidx_train_status(train_id,status);-- 优化查询SELECTidFROMseatsWHEREtrain_id=100ANDstatus=0LIMIT1FORUPDATE;-- 仅锁 1 行
▶ 方案 2:缩短事务时长
  • 两阶段提交
    // 阶段1:锁定座位(短事务)$pdo->beginTransaction();$stmt=$pdo->prepare("SELECT id FROM seats WHERE train_id=? AND status=0 LIMIT 1 FOR UPDATE");$stmt->execute([$trainId]);$seatId=$stmt->fetchColumn();if($seatId){$pdo->exec("UPDATE seats SET status=1 WHERE id=?",[$seatId]);// 标记为已锁定}$pdo->commit();// 阶段2:异步处理支付(无锁)if($seatId){dispatchPaymentJob($seatId);// 放入队列}
▶ 方案 3:无锁设计(终极方案)
  • 预分配座位池
    -- 余票计数表CREATETABLEtrain_inventory(train_idINTPRIMARYKEY,availableINTNOTNULL);
  • 原子扣减
    // 乐观锁扣减库存$stmt=$pdo->prepare(" UPDATE train_inventory SET available = available - 1 WHERE train_id = ? AND available > 0 ");if($stmt->execute([$trainId])&&$stmt->rowCount()>0){// 分配具体座位(无锁)assignSeat($trainId);}
▶ 方案 4:Redis Lua 脚本(超高并发)
  • Lua 脚本保证原子性
    -- check_and_lock.lualocalavailable=redis.call('GET',KEYS[1])iftonumber(available)>0thenredis.call('DECR',KEYS[1])return1endreturn0
  • PHP 调用
    $locked=$redis->eval(file_get_contents('check_and_lock.lua'),["train:100:seats"],1);

四、避坑指南

陷阱破局方案
无索引使用 FOR UPDATE必须为 WHERE 字段加联合索引
事务中调用外部 API用两阶段提交分离锁与业务逻辑
盲目增加超时时间优化锁粒度比调大innodb_lock_wait_timeout更有效

五、终极心法

**“FOR UPDATE 不是枷锁,
而是精度的标尺——

  • 当你缩小粒度
    你在释放并发;
  • 当你缩短持有
    你在提升吞吐;
  • 当你拥抱无锁
    你在铸造韧性。

真正的高并发,
始于对锁的敬畏,
成于对细节的精控。”


结语

从今天起:

  1. 所有FOR UPDATE查询必须有索引
  2. 事务内禁止调用外部 API
  3. 超高并发场景优先考虑 Redis 无锁方案

因为最好的并发控制,
不是加更多锁,
而是精准控制每一比特的竞争。

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

React Native鸿蒙版:KeyboardInteractive交互监听

React Native for OpenHarmony 实战:KeyboardInteractive 键盘交互监听详解 摘要 本文深入探讨React Native在OpenHarmony 6.0.0平台上实现键盘交互监听的技术方案。文章详细解析了KeyboardInteractive组件的核心原理、在OpenHarmony 6.0.0 (API 20)环境下的适配策…

作者头像 李华
网站建设 2026/4/23 16:51:17

【Effective Modern C++】第三章 转向现代C++:15. 尽可能使用constexpr

constexpr作用于对象时,是加强版的const;但作用于函数时含义截然不同 —— 它并非简单限定 “返回 const 值” 或 “结果编译期可知”,而是根据调用场景动态适配。 constexpr对象:必然具备const只读属性,且必须由编译期…

作者头像 李华
网站建设 2026/4/23 13:10:25

Java springboot基于微信小程序的漫画小说阅读系统(源码+文档+运行视频+讲解视频)

文章目录 系列文章目录目的前言一、详细视频演示二、项目部分实现截图三、技术栈 后端框架springboot前端框架vue持久层框架MyBaitsPlus微信小程序介绍系统测试 四、代码参考 源码获取 目的 随着移动互联网的普及,漫画小说阅读成为大众喜爱的娱乐方式。本系统基于…

作者头像 李华
网站建设 2026/4/23 13:12:33

基于SpringBoot的定制化设计服务平台系统(源码+lw+部署文档+讲解等)

课题介绍 随着消费升级与个性化需求的崛起,定制化设计服务的市场需求持续增长,但当前行业内存在设计需求与设计师资源对接不畅、服务流程不规范、需求传递不精准、进度跟踪不透明、交易保障不足等问题,既增加了需求方的沟通成本与试错成本&am…

作者头像 李华
网站建设 2026/4/23 13:14:09

基于SpringBoot的建筑工程项目管理系统(源码+lw+部署文档+讲解等)

课题介绍 随着建筑行业的快速发展,建筑工程项目规模不断扩大、流程日益复杂,当前传统的项目管理模式存在流程混乱、信息传递滞后、各部门协同不畅、进度与成本管控不到位、资料归档不规范等问题,既增加了项目管理成本、延长了项目工期&#x…

作者头像 李华
网站建设 2026/4/23 13:16:57

开箱即用!GLM-4.7-Flash镜像详细使用指南

开箱即用!GLM-4.7-Flash镜像详细使用指南 摘要: 你是否经历过这样的场景:下载好大模型,配环境、装依赖、调参数、改配置……折腾两小时,还没打出一句“你好”? GLM-4.7-Flash 镜像不是“又一个需要编译的…

作者头像 李华