我遇到了一个令我困惑的僵局。我在另一篇文章中读到Mysql 死锁解释需要一个不能解决我困惑的答案。这种情况也出现在另一个关于SHOW INNODB STATUS的参考解释页面中。
我也在附加我的实例 - 这有点不同,因为它锁定了一个索引,一次有间隙,一次没有。
但我认为主要问题仍然是相同的:
如果 Transaction(2) 有 lock A,那么 Transaction(1) 正在等待 Lock A,难道 Transaction(2) 再次请求 lock A会死锁吗?
这听起来不对,尽管这是我们在 InnoDb 状态中看到的。当我尝试直接在 MySql 中重新创建时,对不同的事务使用不同的选项卡 - 一切正常,并且没有这样的死锁。我还附上了我徒劳的重建尝试。
我可能在对这种情况的解释中遗漏了一些东西,我将不胜感激。
我正在使用带有 Hibernate 3 的 MySql 5.1。
------------------------
LATEST DETECTED DEADLOCK
------------------------
110918 14:56:36
*** (1) TRANSACTION:
TRANSACTION 0 40261686, ACTIVE 0 sec, process no 1686, OS thread id 1358170432 updating or deleting
mysql tables in use 1, locked 1
LOCK WAIT 9 lock struct(s), heap size 3024, undo log entries 1
MySQL thread id 101203, query id 77147262 localhost 127.0.0.1 operator Updating
update test_table set created='2011-09-18 14:56:28', customer_id=3, ended=null, lead_id=423, message=null, modified='2011-09-18 14:56:36', priority=0, project_id=74, retries=0, started='2011-09-18 14:56:36', status='PROCESS', user_id=2, inquiry_id=1542 where id=1541
*** (1) WAITING FOR THIS LOCK TO BE GRANTED:
RECORD LOCKS space id 0 page no 7148 n bits 952 index `status` of table `test_table` trx id 0 40261686 lock_mode X locks gap before rec insert intention waiting
*** (2) TRANSACTION:
TRANSACTION 0 40261595, ACTIVE 0 sec, process no 1686, OS thread id 1360034112 updating or deleting, thread declared inside InnoDB 499
mysql tables in use 1, locked 1
9 lock struct(s), heap size 3024, undo log entries 2
MySQL thread id 101209, query id 77147276 localhost 127.0.0.1 operator Updating
update test_table set created='2011-09-18 14:53:22', customer_id=3, ended=null, lead_id=401, message='', modified='2011-09-18 14:56:36', priority=0, project_id=74, retries=3, started='2011-09-18 14:55:46', status='PENDING', user_id=2, inquiry_id=1474 where id=1473
*** (2) HOLDS THE LOCK(S):
RECORD LOCKS space id 0 page no 7148 n bits 952 index `status` of table `test_table` trx id 0 40261595 lock_mode X
*** (2) WAITING FOR THIS LOCK TO BE GRANTED:
RECORD LOCKS space id 0 page no 7148 n bits 952 index `status` of table `test_table` trx id 0 40261595 lock_mode X locks gap before rec insert intention waiting
*** WE ROLL BACK TRANSACTION (1)
尝试重新创建(我删除了一些具有外键的列 - 因为它们在 InnoDb 状态中没有提到):
1) 初始设置:
DROP TABLE IF EXISTS `knowledge`.`aaa`;
CREATE TABLE `knowledge`.`aaa` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`message` longtext,
`status` varchar(16) DEFAULT NULL,
`priority` int(11) NOT NULL DEFAULT '0',
`retries` int(8) NOT NULL DEFAULT '0',
`modified` datetime DEFAULT NULL,
`customer_id` int(11) unsigned NOT NULL DEFAULT '1',
PRIMARY KEY (`id`),
KEY `status` (`status`),
KEY `customer_id` (`customer_id`) -- ,
-- CONSTRAINT `aaa_customer_fk` FOREIGN KEY (`customer_id`) REFERENCES `customer` (`id`),
) ENGINE=InnoDB AUTO_INCREMENT=515 DEFAULT CHARSET=utf8;
insert into aaa
values (1, '', 'AAA', 0, 0, null, 1);
insert into aaa
values (2, '', 'AAA', 0, 0, null, 1);
insert into aaa
values (3, '', 'AAA', 0, 0, null, 1);
2)事务2:开始事务(我使用gui)大概是获取事务2的第一把锁
select * from aaa where status = 'AAA' for update;
3)事务1:开始事务大概是在等待锁——顺便说一句,事务确实在等待。
insert into aaa
values (10, '', 'BBB', 0, 0, null, 1);
4)事务2:大概试图再次等待同一个锁 - 顺便说一句,这根本不等待。
update aaa set status = 'BBB' where id = 1;
update aaa set status = 'BBB' where id = 2;
现在我可以提交事务 2,然后是事务 1,一切正常……没有重现死锁情况……