수정되지 않은 행에서 MySQL 트리거가 실행되지 않도록 방지
MySQL 트리거는 삽입, 업데이트, 삭제 등 테이블에서 특정 작업이 발생할 때 실행됩니다. 그러나 기본적으로 "업데이트 후" 트리거는 테이블 행이 변경되지 않은 경우에도 활성화됩니다.
변경 사항이 없는데도 "업데이트 후" 트리거가 실행되는 이유는 무엇입니까?
업데이트된 행에서 변경 사항을 감지하는 일반적인 방법은 행의 '새' 버전과 '이전' 버전 간의 개별 열 값을 비교하는 것입니다. 그러나 테이블의 열 수가 증가하면 이 접근 방식은 지루해지고 오류가 발생하기 쉽습니다.
해결책: 행 타임스탬프를 활용하여 변경 사항 식별
이 문제를 해결하는 깔끔한 방법은 행 타임스탬프 열을 사용하는 것입니다. MySQL은 각 행에 대해 생성 타임스탬프와 업데이트 타임스탬프라는 두 개의 타임스탬프 열을 유지합니다. 이러한 타임스탬프는 행이 삽입되거나 업데이트될 때마다 자동으로 업데이트됩니다.
타임스탬프 비교를 기반으로 트리거 실행
새 타임스탬프와 기존 타임스탬프를 비교하여 행이 변경되었는지 확인할 수 있습니다. 다음 트리거 코드는 이 접근 방식을 보여줍니다.
<code class="language-sql">CREATE TRIGGER ins_sum AFTER UPDATE ON foo FOR EACH ROW BEGIN IF NEW.ts <> OLD.ts THEN -- 已进行更改;执行触发器主体 END IF; END;</code>
이 트리거에서 "NEW.ts"와 "OLD.ts"는 각각 새로운 타임스탬프 값과 이전 타임스탬프 값을 나타냅니다. 값이 다르면 변경이 발생하고 트리거 본문이 실행됩니다.
예
다음 예를 고려해 보세요.
<code class="language-sql">CREATE TABLE foo (a INT, b INT, ts TIMESTAMP); -- 插入一些行 INSERT INTO foo (a, b) VALUES (1, 1); INSERT INTO foo (a, b) VALUES (2, 2); INSERT INTO foo (a, b) VALUES (3, 3); -- 触发器定义 DELIMITER /// CREATE TRIGGER ins_sum AFTER UPDATE ON foo FOR EACH ROW BEGIN IF NEW.ts <> OLD.ts THEN INSERT INTO bar (a, b) VALUES (NEW.a, NEW.b); END IF; END; /// DELIMITER ; -- 不更改地更新 UPDATE foo SET b = 3 WHERE a = 3; -- 更改地更新 UPDATE foo SET b = 4 WHERE a = 3;</code>
첫 번째 업데이트를 수행한 후에는(변경 사항 없이) 트리거가 실행되지 않습니다. 그러나 두 번째 업데이트에서는 행이 변경되고 트리거가 실행되어 "막대" 테이블에 새 행이 삽입됩니다.
이 솔루션을 사용하면 개별 열 값을 명시적으로 비교할 필요가 없으며 행이 실제로 변경될 때만 트리거가 실행됩니다.
위 내용은 변경되지 않은 행에서 MySQL 'AFTER UPDATE' 트리거가 실행되는 것을 방지하려면 어떻게 해야 합니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!