一次代码报错的分析:MySQLTransactionRollbackException: Lock wait timeout exceeded; try restarting transaction

8 篇文章 0 订阅

1. 报错信息

### Cause: com.mysql.cj.jdbc.exceptions.MySQLTransactionRollbackException: Lock wait timeout exceeded; try restarting transaction
; SQL []; Lock wait timeout exceeded; try restarting transaction; nested exception is com.mysql.cj.jdbc.exceptions.MySQLTransactionRollbackException: Lock wait timeout exceeded; try restarting transaction

2. 报错原因

  1. 两个事物之间出现死锁,导致另外一个事物超时
  2. 某一种表频繁被锁表,导致其他事物无法拿到锁,导致事物超时

3. 问题过程

  1. 查看数据库当前的进程,看一下有无正在执行的慢SQL记录线程。
mysql> show  processlist;
  1. 查看当前的事务
当前运行的所有事务:
mysql> SELECT * FROM information_schema.INNODB_TRX;
当前出现的锁
mysql> SELECT * FROM information_schema.INNODB_LOCKs;
锁等待的对应关系
mysql> SELECT * FROM information_schema.INNODB_LOCK_waits;
  1. 解释:看事务表INNODB_TRX,里面是否有正在锁定的事务线程,看看ID是否在show processlist里面的sleep线程中,如果是,就证明这个sleep的线程事务一直没有commit或者rollback而是卡住了,我们需要手动kill掉。
    搜索的结果是在事务表发现了很多任务,这时候最好都kill掉。

  2. 批量删除事务表中的事务
    我这里用的方法是:通过information_schema.processlist表中的连接信息生成需要处理掉的MySQL连接的语句临时文件,然后执行临时文件中生成的指令。

mysql>  select concat('KILL ',id,';') from information_schema.processlist where user='cms_bokong';
+------------------------+
| concat('KILL ',id,';') |
+------------------------+
| KILL 10508;            |
| KILL 10521;            |
| KILL 10297;            |
+------------------------+
18 rows in set (0.00 sec)

当然结果不可能只有3个,这里我只是举例子。参考链接上是建议导出到一个文本,然后执行文本。而我是直接copy到记事本处理掉 ‘|’,粘贴到命令行执行了。都可以。

kill掉以后再执行SELECT * FROM information_schema.INNODB_TRX; 就是空了。

这时候系统就正常了

故障排查

mysql都是autocommit配置
mysql> select @@autocommit;
+--------------+
| @@autocommit |
+--------------+
|            0 |
+--------------+
1 row in set (0.00 sec)

参考:MySQL事务锁问题-Lock wait timeout exceeded


4. 问题解决

经过了解业务信息,一次性执行500次的UPDATE操作,一次UPDATE耗时100毫秒,导致表被锁50秒。

为被更新操作增加索引之后,此问题解决,执行速度加快。

大量update语句报错MySQLTransactionRollbackException: Lock wait timeout exceeded; try restarting transaction是因为多个事务同时对相同的行进行操作,导致行锁冲突。当一个事务在操作某一行时,另一个事务也需要对同一行进行操作时,会发生锁等待超时的情况。这种情况下,MySQL会自动回滚超时的事务,并抛出MySQLTransactionRollbackException异常。 为了解决这个问题,可以采取以下方法来避免锁等待超时的情况: 1. 提高事务超时时间:可以通过设置innodb_lock_wait_timeout参数来增加事务超时时间,以便给事务更多的时间来等待锁释放。例如,可以执行"SET GLOBAL innodb_lock_wait_timeout=1500;"来将超时时间设置为1500秒。 2. 优化事务并发控制:可以合理设计事务的并发控制策略,避免多个事务同时对相同的行进行操作。可以使用事务锁定级别、行级锁、表级锁等机制来控制事务的并发访问。同时,也可以考虑对数据库的表结构进行优化,减少事务冲突的可能性。 3. 检查代码逻辑:可能是代码逻辑中存在问题,导致多个事务同时对相同的行进行操作。可以仔细检查代码,确保在操作同一行时,只有一个事务在进行操作。 需要注意的是,锁等待超时可能是由于数据库负载过高、事务处理时间过长等原因导致的。因此,除了以上方法外,还需要考虑优化数据库性能和调整系统配置来解决问题。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [MySQLTransactionRollbackException: Lock wait timeout exceededtry restarting transaction异常解决](https://blog.csdn.net/qq_24309787/article/details/131132586)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* *3* [MySQLTransactionRollbackException: Lock wait timeout exceededtry restarting transaction](https://blog.csdn.net/run_boy_2022/article/details/130006997)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值