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;