드문 교착 상태 오류가 발생했습니다. 두 쿼리 작업이 서로의 결과에 의존 할 때 데드락이 발생하여 MySQL이 그 중 하나를 롤백 할 때 교착 상태가 발생한다는 것을 이해합니다. 하지만 내 상황에서 MySQL은 자동 커밋 모드이고 트리거를 발생시키는 새로운 레코드를 삽입하고 있습니다. 그래서 나는 죽은 자물쇠 상황을 일으키는 이유를 얻지 못합니다. 여기 MySQL 트리거를 발생시키는 인서트로 교착 상태가 발생했습니다.
내 테이블 스키마입니다 :---- 사용자 표 ----
CREATE TABLE `users` (
`insta_id` bigint(20) unsigned NOT NULL,
`name` varchar(50) NOT NULL,
`password` varchar(60) NOT NULL,
`gem` int(10) unsigned DEFAULT '20',
`coin` int(10) unsigned DEFAULT '20',
PRIMARY KEY (`insta_id`),
UNIQUE KEY `insta_id` (`insta_id`)
) ENGINE=InnoDB DEFAULT CHARSET=latin1
--- 표를 like_requests ---
CREATE TABLE `like_requests` (
`req_id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
`insta_id` bigint(20) unsigned NOT NULL,
`media_id` varchar(50) NOT NULL,
`remaining_like` int(10) unsigned NOT NULL,
`active` tinyint(1) NOT NULL DEFAULT '1',
`count` int(10) unsigned NOT NULL,
PRIMARY KEY (`req_id`),
KEY `insta_id` (`insta_id`),
KEY `media_id` (`media_id`),
CONSTRAINT `like_requests_ibfk_1` FOREIGN KEY (`insta_id`) REFERENCES `users`(`insta_id`)
) ENGINE=InnoDB AUTO_INCREMENT=103902 DEFAULT CHARSET=latin1
--- 좋아 표 ---
CREATE TABLE `likes` (
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
`insta_id` bigint(20) unsigned NOT NULL,
`media_id` varchar(50) NOT NULL,
`req_id` bigint(20) unsigned DEFAULT NULL,
`date` timestamp NULL DEFAULT CURRENT_TIMESTAMP,
PRIMARY KEY (`id`),
UNIQUE KEY `id` (`id`),
KEY `req_id` (`req_id`),
KEY `insta_id` (`insta_id`),
KEY `media_id` (`media_id`),
CONSTRAINT `likes_ibfk_1` FOREIGN KEY (`req_id`) REFERENCES `like_requests`(`req_id`),
CONSTRAINT `likes_ibfk_2` FOREIGN KEY (`insta_id`) REFERENCES `users`(`insta_id`)
) ENGINE=InnoDB AUTO_INCREMENT=1704209 DEFAULT CHARSET=latin1
나는 방아쇠를 가지고있다. 몇 가지 간단한 삽입을하고 함께
CREATE TRIGGER `after_insert_likes` AFTER INSERT ON `likes`
FOR EACH ROW BEGIN
UPDATE users SET users.coin=users.coin+1
WHERE users.insta_id = NEW.insta_id LIMIT 1;
IF NEW.req_id IS NOT NULL THEN
UPDATE like_requests
SET like_requests.remaining_like = like_requests.remaining_like-1
WHERE like_requests.req_id = NEW.req_id
AND like_requests.remaining_like > 0
LIMIT 1;
END IF;
END
: N은 다음과 같이 정의, 테이블을 좋아
$sql = "INSERT INTO likes (insta_id,media_id,req_id) VALUES (?,?,?);";
$pdo = $this->db;
$statement = $pdo->prepare($sql);
$statement->bindValue(1,$data['id'],PDO::PARAM_INT);
$statement->bindValue(2,$data['media_id']);
$statement->bindValue(3,$data['req_id'],PDO::PARAM_INT);
try
{
$statement->execute();
return GetOkResponseWithMessage($response,"Like was submitted");
}
catch (PDOException $exc)
{
return GetErrorResponseWithMessage($response,$exc->getMessage(),500);
}
나는 다음과 같은 죽은 잠금 오류 로그를 얻을 :
*** (1) TRANSACTION:
TRANSACTION 29031910, ACTIVE 1 sec starting index read
mysql tables in use 4, locked 4
LOCK WAIT 7 lock struct(s), heap size 1184, 3 row lock(s), undo log entries 1
MySQL thread id 264238, OS thread handle 0x7f6522c6eb00, query id 753506 localhost xxxx updating
UPDATE users SET users.coin=users.coin+1 WHERE users.insta_id=NEW.insta_id LIMIT 1
*** (1) WAITING FOR THIS LOCK TO BE GRANTED:
RECORD LOCKS space id 14 page no 1560 n bits 128 index `PRIMARY` of table `insta_star`.`users` trx table locks 4 total table locks 4 trx id 29031910 lock_mode X locks rec but not gap waiting lock hold time 0 wait time before grant 0
*** (2) TRANSACTION:
TRANSACTION 29031909, ACTIVE 1 sec starting index read
mysql tables in use 4, locked 4
7 lock struct(s), heap size 1184, 3 row lock(s), undo log entries 1
MySQL thread id 264237, OS thread handle 0x7f65209f8b00, query id 753507 localhost xxxx updating
UPDATE users SET users.coin=users.coin+1 WHERE users.insta_id=NEW.insta_id LIMIT 1
*** (2) HOLDS THE LOCK(S):
RECORD LOCKS space id 14 page no 1560 n bits 128 index `PRIMARY` of table `insta_star`.`users` trx table locks 4 total table locks 4 trx id 29031909 lock mode S locks rec but not gap lock hold time 0 wait time before grant 0
*** (2) WAITING FOR THIS LOCK TO BE GRANTED:
RECORD LOCKS space id 14 page no 1560 n bits 128 index `PRIMARY` of table `insta_star`.`users` trx table locks 4 total table locks 4 trx id 29031909 lock_mode X locks rec but not gap waiting lock hold time 0 wait time before grant 0
*** WE ROLL BACK TRANSACTION (2)
해야하는까지하지이를 죽은 자물쇠 대신에 자물쇠 대기에서?
트랜잭션을 다시 시작하지 않고 어떻게이 문제를 극복 할 수 있습니까?
트리거 된 삽입을 추가 할 수 있습니까? 트랜잭션 주위를 감싸는 경우에는 full bock을 게시하십시오 – e4c5
@ e4c5 삽입 코드는 단일 쿼리로 실행됩니다. 삽입을 수행하는 PHP 코드를 추가했습니다. – BlackBrain