我在表格上有一些触发器BEFORE INSERT
,AFTER DELETE
。如何确定触发器是否失败,然后我的查询会回滚?
我的意思是我想确定,查询和触发器都可以工作,或者它们都不起作用。触发事务也是如此吗?
答案 0 :(得分:3)
对于事务表,语句失败应导致回滚语句执行的所有更改。触发器失败会导致语句失败,因此触发器失败也会导致回滚。
答案 1 :(得分:1)
我可以通过存储过程显示这个。这个概念从这个answer取消了wchiquito。我相信你会发现这是一个更详尽的答案。这只是一个例子。根据您的特定需求(其他触发器类型)进行必要的更改。如何执行不使用存储过程的外部的mysql触发器信号是任何人的猜测。因此,如果您不愿意或无法执行存储过程,不再阅读。
请注意,为方便起见,我们会留下任何掉落或截断的内容。
create database trigTranTest; -- creates a separate database for testing
use trigTranTest; -- use that database
-- drop table tableA;
create table tableA
( id int auto_increment primary key,
something varchar(100) not null,
age int not null, -- do not accept unlucky 13
myDT datetime not null
);
-- drop table tableB;
create table tableB
( -- simply to demonstrate multiple tables in a transaction and that they are honored as a group (ie: Transaction)
-- all or nothing basically
id int auto_increment primary key,
something varchar(100) not null,
myDT datetime not null
);
-- drop table auditInfoNotInTrans;
create table auditInfoNotInTrans
( -- a boring table outside of Transaction to show an attempt was made
id int auto_increment primary key,
debugInfo varchar(100) not null,
myDT datetime not null
);
-- POINT A
drop trigger if exists tableA_BeforeIns;
DELIMITER $$
create trigger tableA_BeforeIns before insert on tableA
for each row
begin
if new.age = 13 then
-- disallow unlucky age=13 for inserts. Wait another year.
signal sqlstate '45000' set message_text = "tableA_BeforeIns bombed due to age=13";
end if;
end$$
DELIMITER ;
-- POINT B
关于触发器的快速说明:如果您尝试插入年龄= 13,则将设置信号。这将启动交易的最终ROLLBACK
。
请注意DELIMITERS非常重要。要修改上述内容,请突出显示POINT A
和POINT B
之间的所有文字并执行。该块将使用DELIMITER后方需要的那种疼痛来执行跌落和娱乐。如果没有DELIMITER,错误1064 即将来临。翻译:它不会起作用。哪些不起作用?创建触发器的部分。
-- POINT A
drop procedure if exists insertChunk;
DELIMITER $$
CREATE PROCEDURE insertChunk(pSomething varchar(100), pAge int)
-- takes two parameters, a string for a thing, and an age
BEGIN
-- idea lifted from https://stackoverflow.com/a/19908197 by user wchiquito
-- so spread the appreciation there
DECLARE `_rollback` BOOL DEFAULT 0;
DECLARE CONTINUE HANDLER FOR SQLEXCEPTION SET `_rollback` = 1;
-- the following happens outside of the Transaction
insert auditInfoNotInTrans(debugInfo,myDT) values(pSomething,now());
-- now our Transaction part begins
START TRANSACTION;
insert tableA(something,age,myDT) values (pSomething,pAge,now()); -- pAge being unlucky 13 fails via the Trigger
IF `_rollback` THEN
ROLLBACK;
ELSE
insert tableB(something,myDT) values (pSomething,now());
COMMIT;
END IF;
END$$
DELIMITER ;
-- POINT B
快速说明:START TRANSACTION
发生后,我们将转到COMMIT
,除非我们的触发器发出SQLSTATE信号,从而导致ROLLBACK
。
如前所述,突出显示并执行POINT A
和POINT B
内的所有代码,以对上述内容进行修改。这次是存储过程,但类似于之前的create trigger
。
含义,类似于使用DELIMITER块的安全包装进行的触发器修改。否则,错误1064即将发生,并且不会创建存储过程。
请注意,在测试过程中,为了您的方便,会留下以下已删除的截断值。
-- truncate tableA;
-- truncate tableB;
-- truncate auditInfoNotInTrans;
call insertChunk('frog',1);
call insertChunk('lizard',13); -- force a Trigger failure with the unlucky 13
call insertChunk('snake',2);
select * from auditInfoNotInTrans;
+----+-----------+---------------------+
| id | debugInfo | myDT |
+----+-----------+---------------------+
| 1 | frog | 2016-06-10 15:09:02 |
| 2 | lizard | 2016-06-10 15:09:06 |
| 3 | snake | 2016-06-10 15:09:08 |
+----+-----------+---------------------+
select * from tableA;
+----+-----------+-----+---------------------+
| id | something | age | myDT |
+----+-----------+-----+---------------------+
| 1 | frog | 1 | 2016-06-10 15:09:02 |
| 2 | snake | 2 | 2016-06-10 15:09:08 |
+----+-----------+-----+---------------------+
select * from tableB;
+----+-----------+---------------------+
| id | something | myDT |
+----+-----------+---------------------+
| 1 | frog | 2016-06-10 15:09:02 |
| 2 | snake | 2016-06-10 15:09:08 |
+----+-----------+---------------------+
结果符合预期,表彰交易处理,不允许年龄= 13的插入。当然,它是任意的,但我们必须以某种方式测试它。
最后一个视觉效果。直接从Mysql Workbench运行插入,年龄= 13
insert tableA(something,age,myDT) values ('turtle',13,now());
错误代码:1644。tableA_Before由于年龄= 13 0.000秒而被轰炸
drop database trigTranTest;
测试数据库已被删除并且已经消失。