我可以在MySQL触发器中使用类似事务的功能

时间:2012-02-13 04:27:01

标签: mysql transactions triggers

我有一个插入触发器,它从表A中的行中获取一组列值,并在表B中插入一些列值并保留在表C中。我需要此操作成为一个事务,其中如果数据有一些错误如果插入表B而不是C,则应回滚整个插入操作。

我研究了手册,它在this页的最后一页说明触发器中不允许交易

有没有办法在mysql中实现我想要的东西。

2 个答案:

答案 0 :(得分:4)

是的,你可以,但你如何做到这取决于你的版本。

首先,触发器本身是事务性的;在您的情况下,您有一个插入触发器,执行两次进一步插入。如果其中一个失败,您将获得所需的效果。

考虑以下示例:

CREATE TABLE a (colA INT);
CREATE TABLE b (colB INT);
CREATE TABLE c (colC INT);
delimiter :
CREATE TRIGGER testtrig BEFORE INSERT ON a
  FOR EACH ROW BEGIN
    INSERT INTO b(colB) VALUES(NEW.colA);
    INSERT INTO c(banana) VALUES (NEW.colA); -- note the faulty column name
END;:
delimiter ;

现在,当我运行一个失败的插入时,会发生这种情况:

mysql> INSERT INTO a VALUES (5);
ERROR 1054 (42S22): Unknown column 'banana' in 'field list'
mysql> SELECT * FROM a;
Empty set (0.00 sec)

这符合您想要的结果。

更一般地说,如果你有逻辑可以在尝试插入之前验证你的数据,你可以用不同的方式使触发失败:

  • 在MySQL 5.5中,您可以使用SIGNAL机制从触发器引发错误,从而导致整个插入失败。
  • 在MySQL 5.5之前,您可以生成故意错误以使触发器失败。

我猜你正在使用你问题中的链接5.0,所以如果你需要,你可以执行故意的错误,例如故意插入无效列,以使触发失败。但是,您在问题中描述的情况已经在交易中处理,如我的答案开头所述。

答案 1 :(得分:0)

默认情况下,您得到了所要求的内容 - 触发器中的任何错误都会导致语句失败。因此,如果语句上有事务,则会在该语句之前回滚数据。如果没有交易,那么你就没有。

这可能是触发器中不允许创建或结束事务的原因。

因此不需要存储过程。实际上,如果尝试创建事务,则从触发器调用的存储过程可能会导致错误。

但在执行导致触发的操作之前,请随意使用存储过程启动事务。