Lock wait timeout exceeded; try restarting transaction

时间:2022-04-01 18:21:47

mysql出现:Lock wait timeout exceeded; try restarting transaction

什么问题导致的呢?绝对是程序的问题,因为另一个线程锁住了表或者记录导致后来到请求无法完成。

如何产生的:

表A

存储过程B

DELIMITER $$

DROP PROCEDURE IF EXISTS `B` $$
CREATE PROCEDURE `B`( )
BEGIN
SET autocommit=0;

select * from A where id=1 for update;

.......
commit;
END $$

DELIMITER ;

 

存储过程C

DELIMITER $$

DROP PROCEDURE IF EXISTS `C` $$
CREATE PROCEDURE `C`( )
BEGIN
select * from A where id=1 for update;

........
END $$

DELIMITER ;

 

 

当调用完B以后(autocommit=0),调用c(此时该记录被锁定,因为没有commit该锁就一直在,造成了全局的死锁.这里其实还是代码问题,因为在autocommit=1的情况下for update 实际上是不起作用的),此后无论任何地方再来锁A表的记录的时候就会出这个超时的异常。

autocommit是connection级别的变量,但是锁确实全局的(select @@autocommit;),在我们使用数据库连接池的情况下,connection.close()其实只关闭了所有的statement,

所以当同一个connection调用了B以后再去调用C.造成了一个全局的锁。

如果只是想加锁的范围在特定的语句,则应该使用START TRANSACTION语句:

START TRANSACTION;
select * from A where id=1 for update;
。。。
COMMIT;