数据库连接未释放导致表被锁问题排查与处理

🚨 问题现象

  • 执行 UPDATE / DELETE / ALTER TABLE 等操作时 长时间无响应 或报错:

    复制代码
    Lock wait timeout exceeded; try restarting transaction

    复制代码
    Table 'xxx' is locked
  • 应用写入/更新功能卡住,但读操作可能正常。

  • 怀疑有数据库连接未正确关闭,导致事务未提交或表级锁未释放。


🔍 问题原因分析

  • 某个应用程序(或调试脚本)开启了数据库事务但未提交或回滚
  • 连接未被显式关闭,连接池未回收 ,导致:
    • 行锁 / 表锁持续持有;
    • 后续操作因等待锁而阻塞;
  • 常见于:
    • 开发环境手动执行 BEGIN; 后忘记 COMMIT/ROLLBACK
    • 应用代码中异常路径未关闭连接(连接泄漏);
    • 长时间运行的查询未结束。

💡 关键点:在 InnoDB 中,未提交的事务会持有行锁;在 MyISAM 中,写操作会直接加表锁。


🛠️ 排查步骤(以 MySQL 为例)

1. 登录数据库(使用高权限账号)

复制代码
mysql -u root -p

即使连接池满,MySQL 通常仍允许一个 SUPER 用户登录。

2. 查看当前活跃连接和锁等待情况

复制代码
-- 查看所有连接
SHOW PROCESSLIST;

-- 更详细信息(推荐)
SELECT 
  id, user, host, db, command, time, state, info 
FROM information_schema.processlist 
ORDER BY time DESC;

重点关注:

  • Time:执行/空闲时长(秒)
  • State:如 Locked, Sending data, Sleep
  • Info:正在执行的 SQL(若为 NULLCommand=Sleep,说明是空闲连接)

3. 检查是否有未提交的事务(InnoDB)

复制代码
-- 查看 InnoDB 引擎状态(含锁信息)
SHOW ENGINE INNODB STATUS\G

在输出中查找 TRANSACTIONSLOCKS WAITING 部分,可看到:

  • 哪些事务在等待锁;
  • 哪些事务持有锁;
  • 对应的连接 ID(thread ID)。

4. 定位"可疑"连接

例如,发现一条连接:

  • User: dev_user
  • Host: 192.168.1.100:54321
  • Time: 3600(已空闲1小时)
  • Info: NULL
  • Command: Sleep

→ 极可能是未关闭的调试连接,且可能持有未提交事务。


✅ 解决方法:终止无用连接

终止指定连接(释放锁)

复制代码
KILL <connection_id>;
-- 示例
KILL 12345;

⚠️ 使用 KILL(不是 KILL QUERY),确保整个连接断开,事务自动回滚,锁被释放。

批量清理(谨慎使用)

复制代码
-- 生成 kill 语句(先预览)
SELECT CONCAT('KILL ', id, ';') 
FROM information_schema.processlist 
WHERE user = 'dev_user' 
  AND command = 'Sleep' 
  AND time > 600;

复制结果,在新窗口执行。


✅ 验证是否恢复

  • 再次执行之前卡住的 SQL,确认能正常完成;
  • 检查 SHOW PROCESSLIST; 中不再有长时间空闲或锁定状态的连接。

🛡️ 预防措施

层面 措施
开发习惯 手动测试 SQL 时,避免只写 BEGIN; 而不提交;用完及时关闭客户端连接
应用代码 确保使用 try-finally 或连接池自动管理(如 HikariCP 的 close()
数据库配置 设置合理的超时: SET GLOBAL wait_timeout = 600; SET GLOBAL interactive_timeout = 600;
监控告警 监控 Threads_connectedInnodb_row_lock_waits 指标

💡 经验总结

  • 表锁/行锁问题往往源于"活着但无用"的连接,而非 SQL 本身性能问题;
  • SHOW PROCESSLIST + SHOW ENGINE INNODB STATUS 是排查锁问题的黄金组合;
  • KILL 连接是快速恢复业务的有效手段,但需定位准确,避免误杀;
  • 根本解决在于规范开发流程 + 健壮的连接管理

📌 附:常用命令速查

复制代码
-- 查看连接
SHOW PROCESSLIST;

-- 查看 InnoDB 锁状态
SHOW ENGINE INNODB STATUS\G

-- 终止连接
KILL 12345;

-- 查看超时设置
SHOW VARIABLES LIKE '%timeout%';
相关推荐
知识分享小能手几秒前
Redis入门学习教程,从入门到精通,Redis服务配置知识点详解(3)
数据库·redis·学习
q5431470872 分钟前
mybatis plus打印sql日志
数据库·sql·mybatis
姜太小白7 分钟前
【SQLServer】SQL Server 2022 连接证书错误解决
网络·数据库·sqlserver
良许Linux15 分钟前
FPGA的选型和应用
数据库·图像处理·计算机视觉·fpga开发
JZC_xiaozhong25 分钟前
医疗器械行业供应商主数据管理:多视角下的破局与重构
大数据·数据库·重构·数据分析·mdm·主数据管理·数据集成与应用集成
yunyun3212328 分钟前
用Python监控系统日志并发送警报
jvm·数据库·python
学编程的小程1 小时前
筑牢数据安全防线——金仓数据库SQL防火墙硬核防护解析
数据库·sql
ReSearch1 小时前
工业物联网的“瘦身”革命:Go 实现 20MB 级边缘存储,基于 LSM-Tree 的深度定制实践
数据库·go
TDengine (老段)1 小时前
TDengine IDMP 组态面板 —— 锚点
大数据·数据库·物联网·时序数据库·tdengine·涛思数据
Navicat中国1 小时前
Navicat 模式设计全解:解决数据库开发 3 大核心痛点
数据库·数据库开发·navicat·模式设计