一次因缺失索引引发的线上锁超时事故

前言

周四深夜,万籁俱寂,刚要入睡,手机突然响起。

"线上列表查询接口返回空数据,群里已经炸锅了------大量用户反馈 APP 无法正常使用,赶紧排查!"

听到这句话,我瞬间清醒,睡意全无,立刻从床上弹起。

问题定位

匆忙打开电脑,连上生产环境,首先检查 MySQL 的运行状态。观察到 CPU 和内存使用率都很低,基本可以排除数据库因负载过高导致响应异常的可能。

接着查看应用的错误日志,发现大量如下异常:

java 复制代码
Cause: com.mysql.cj.jdbc.exceptions.MySQLTransactionRollbackException: 
Lock wait timeout exceeded; try restarting transaction

nested exception is com.mysql.cj.jdbc.exceptions.MySQLTransactionRollbackException: 
Lock wait timeout exceeded; try restarting transaction

显然,问题出在锁等待超时上。于是,我立即执行以下 SQL 查询,定位造成阻塞的具体事务:

sql 复制代码
SELECT  
    dl.REQUESTING_ENGINE_TRANSACTION_ID AS waiting_trx_id,  
    dl.BLOCKING_ENGINE_TRANSACTION_ID AS blocking_trx_id,  
    l2.OBJECT_NAME AS table_name,  
    l1.LOCK_DATA AS blocked_by_data,  
    t1.trx_query AS waiting_sql,  
    t2.trx_query AS blocking_sql,  
    t2.trx_started AS blocking_started  
FROM performance_schema.data_lock_waits dl  
JOIN performance_schema.data_locks l2  
  ON dl.REQUESTING_ENGINE_LOCK_ID = l2.ENGINE_LOCK_ID  
JOIN performance_schema.data_locks l1  
  ON dl.BLOCKING_ENGINE_LOCK_ID = l1.ENGINE_LOCK_ID  
LEFT JOIN information_schema.innodb_trx t1  
  ON dl.REQUESTING_ENGINE_TRANSACTION_ID = t1.trx_id  
LEFT JOIN information_schema.innodb_trx t2  
  ON dl.BLOCKING_ENGINE_TRANSACTION_ID = t2.trx_id;

查询结果清晰地显示出一条正在等待锁的 SQL(见下图):

顺着这条 SQL,我迅速定位到对应的业务代码。问题根源浮出水面:一个未加索引条件的长事务执行了 UPDATE 操作

根本原因分析

该表数据量超过 90 万行,且除主键外没有任何二级索引。在 InnoDB 存储引擎下,执行如下语句:

sql 复制代码
UPDATE table SET col = val WHERE non_indexed_col = xxx;

由于 WHERE 条件字段无索引,MySQL 不得不进行全表扫描 。在可重复读(RR)隔离级别下,InnoDB 会对扫描到的每一行加记录锁 ,甚至可能加上间隙锁(Gap Lock) 。结果就是------整张表几乎被锁住,其他并发事务在尝试获取锁时纷纷超时,最终导致接口无数据返回、APP 功能大面积不可用。

解决方案与最佳实践

为避免类似问题再次发生,我们总结了以下几点优化建议:

  • 避免长事务:将大范围更新拆分为小批量操作(如分页更新),缩短事务持有时间。
  • 确保 WHERE 条件字段有索引:防止全表扫描引发的行锁膨胀。
  • 及时提交事务:减少锁资源的占用时长,提升系统并发能力。

这次深夜"救火"经历再次提醒我们:在高并发场景下,一个看似微小的索引缺失,可能就是压垮系统的最后一根稻草

相关推荐
一 乐1 天前
婚纱摄影网站|基于ssm + vue婚纱摄影网站系统(源码+数据库+文档)
前端·javascript·数据库·vue.js·spring boot·后端
码事漫谈1 天前
Protocol Buffers 编码原理深度解析
后端
码事漫谈1 天前
gRPC源码剖析:高性能RPC的实现原理与工程实践
后端
踏浪无痕1 天前
AI 时代架构师如何有效成长?
人工智能·后端·架构
程序员小假1 天前
我们来说一下无锁队列 Disruptor 的原理
java·后端
武子康1 天前
大数据-209 深度理解逻辑回归(Logistic Regression)与梯度下降优化算法
大数据·后端·机器学习
maozexijr1 天前
Rabbit MQ中@Exchange(durable = “true“) 和 @Queue(durable = “true“) 有什么区别
开发语言·后端·ruby
源码获取_wx:Fegn08951 天前
基于 vue智慧养老院系统
开发语言·前端·javascript·vue.js·spring boot·后端·课程设计
独断万古他化1 天前
【Spring 核心: IoC&DI】从原理到注解使用、注入方式全攻略
java·后端·spring·java-ee
毕设源码_郑学姐1 天前
计算机毕业设计springboot基于HTML5的酒店预订管理系统 基于Spring Boot框架的HTML5酒店预订管理平台设计与实现 HTML5与Spring Boot技术驱动的酒店预订管理系统开
spring boot·后端·课程设计