UPDATE多导致索引变慢,因需同步维护B+树索引,引发页分裂、刷脏、唯一校验及锁争抢;应优化索引设计、启用change_buffering、避免全表扫描更新、合理使用upsert与锁策略。为什么 UPDATE 一多,索引就变慢?因为 MySQL 在更新带索引的字段时,不仅改数据行,还要同步维护 B+ 树索引结构------尤其是二级索引,每次 UPDATE 都可能触发页分裂、缓冲池刷脏、甚至唯一性校验。高并发下这些操作争抢 index latch 和 buffer pool mutex,锁等待直接堆起来。实操建议:避免在 WHERE 条件里用非索引字段更新索引字段(例如 UPDATE t SET status=1 WHERE name='xxx',而 name 没索引)------会全表扫描+逐行更新索引,锁住整张表把高频更新的列和查询条件列拆开:比如 status 经常变,但 created_at 几乎不变,就别把它们塞进同一个联合索引确认 innodb_change_buffering 开启(默认是 all),它能缓存非唯一二级索引的更新,减少随机 IO ------ 但只对离散更新有效,批量顺序写反而可能降低收益INSERT ... ON DUPLICATE KEY UPDATE 的锁范围比你想的大这个语法看着像"存在就改,不存在就插",实际执行时,InnoDB 会对 INSERT 尝试路径上的所有间隙(gap)加 INSERT_INTENTION 锁,并对命中记录加 X 锁。如果唯一索引冲突频繁,很容易卡在间隙锁等待上。常见错误现象:Deadlock found when trying to get lock; try restarting transaction,尤其出现在按时间戳或自增 ID 批量 upsert 场景。实操建议:确保冲突判断字段是 UNIQUE 或 PRIMARY KEY,否则会退化成全表扫描+行锁批量操作时,按主键升序排序后再提交,减少间隙锁交叉(例如先处理 id=100,再 id=200,而不是反过来)若只是想避免重复插入,且不关心是否真更新了,用 INSERT IGNORE 更轻量------它遇到唯一冲突直接跳过,不加 X 锁什么时候该删掉二级索引?不是所有 WHERE 条件都值得建索引。每多一个二级索引,INSERT/UPDATE/DELETE 就得多维护一棵树;更麻烦的是,MySQL 优化器可能因索引太多选错执行计划,反而让 UPDATE 变慢。 AI Code Reviewer AI自动审核代码
相关推荐
数据最前线5 分钟前
亡羊补牢?Oracle 计划推出月度安全补丁X566116 分钟前
CSS Flex布局如何让特定子元素不参与缩放_设置flex-shrink- 0的实战技巧databook18 分钟前
告别手动计算,SymPy 初识与 Manim 联动jayson.h23 分钟前
可视化界面weixin_4440129324 分钟前
CSS如何快速实现网站换肤功能_利用CSS变量重置全局颜色方案kgduu31 分钟前
python中的魔法方法m0_5967490934 分钟前
Vue.js计算属性computed依赖追踪与副作用函数effect关联机制zgdlsz35 分钟前
羲之文化传承人王杰宝:沉厚笔墨间的守正出新打小就很皮...35 分钟前
基于 Python + LangChain + SQL 生成自动查询数据实战xcLeigh38 分钟前
KES大小写混合路径+国产OS/文件系统兼容实战