mysql事务 - 回滚任何异常

Urb*_*leg 31 mysql transactions rollback

如果在mysql命令列表上发生任何错误,是否可以自动回滚?

例如:

begin transaction;

insert into myTable values1 ...
insert into myTable values2 ...;  -- will throw an error

commit;
Run Code Online (Sandbox Code Playgroud)

现在,在执行我希望整个交易失败,因此我应该看到值1在myTable的.但不幸的是,即使事务有错误,表也会被值1傀儡.

我有什么想法让它回滚?(再次,任何错误)?

编辑 - 从DDL更改为标准SQL

wch*_*ito 49

您可以使用13.6.7.2.DECLARE ... HANDLER语法以下列方式:

DELIMITER $$

CREATE PROCEDURE `sp_fail`()
BEGIN
    DECLARE `_rollback` BOOL DEFAULT 0;
    DECLARE CONTINUE HANDLER FOR SQLEXCEPTION SET `_rollback` = 1;
    START TRANSACTION;
    INSERT INTO `tablea` (`date`) VALUES (NOW());
    INSERT INTO `tableb` (`date`) VALUES (NOW());
    INSERT INTO `tablec` (`date`) VALUES (NOW()); -- FAIL
    IF `_rollback` THEN
        ROLLBACK;
    ELSE
        COMMIT;
    END IF;
END$$

DELIMITER ;
Run Code Online (Sandbox Code Playgroud)

有关完整示例,请检查以下SQL小提琴.

  • 谢谢你的回答:).你认为我们为每个新脚本保存一个新的存储过程是个好主意吗?因为我们每个版本有10到15个脚本,所以我们将有数百个存储过程. (3认同)
  • @Xenos不,尽管会浪费时间。如果前两个查询中的任何一个失败,则_rollback仍设置为1,因此该函数将执行ROLLBACK;而不是COMIT;。但是,以下查询仍将执行(在事务内),直到稍后最终回滚。MySQL选择了这种继续出错的行为,这真让我感到困惑。与之相反,PostgreSQL将事务置于失败状态,保证事务中所有将来的查询都会失败(“ ROLLBACK TO”除外),并且在提交时会隐式回滚。 (3认同)

KGs*_*KGs 21

如果您需要在代码中使用SIGNAL特定的SQL EXCEPTION,则可以使用EXIT HANDLER.例如:

DELIMITER $$

CREATE PROCEDURE `sp_fail`()
BEGIN
    DECLARE EXIT HANDLER FOR SQLEXCEPTION
    BEGIN
        ROLLBACK;  -- rollback any changes made in the transaction
        RESIGNAL;  -- raise again the sql exception to the caller
    END;

    START TRANSACTION;
    insert into myTable values1 ...
    IF fail_condition_meet THEN
        SIGNAL SQLSTATE '45000' SET MESSAGE_TEXT = 'Custom error detected.', MYSQL_ERRNO = 2000;
    END IF;
    insert into myTable values2 ...  -- this will not be executed
    COMMIT; -- this will not be executed
END$$

DELIMITER ;
Run Code Online (Sandbox Code Playgroud)


Jam*_*mes 8

上面的解决方案很好,但让它变得更简单

DELIMITER $$

CREATE PROCEDURE `sp_fail`()
BEGIN
    DECLARE EXIT HANDLER FOR SQLEXCEPTION
    BEGIN
        ROLLBACK;  -- rollback any error in the transaction
    END;

    START TRANSACTION;
    insert into myTable values1 ...
    insert into myTable values2 ...  -- Fails
    COMMIT; -- this will not be executed
END$$

DELIMITER ;
Run Code Online (Sandbox Code Playgroud)