web-dev-qa-db-ja.com

mysqlトランザクション-例外があればロールバック

Mysqlコマンドのリストでエラーが発生した場合、自動的にロールバックできますか?

たとえば、次のような行に沿ったものです。

begin transaction;

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

commit;

今、実行時にトランザクション全体が失敗するようにしたいので、[〜#〜] not [〜#〜] myTableのvalues1を参照してください。しかし、残念ながら、トランザクションにエラーがある場合でも、テーブルにはvalues1が挿入されます。

ロールバックする方法はありますか? (再度、エラーが発生した場合)?

編集-DDLから標準SQLに変更

29
Urbanleg

13.6.7.2。DECLARE ... HANDLER Syntax を次の方法で使用できます。

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 ;

完全な例については、次の SQL Fiddle を確認してください。

46
wchiquito

たとえば、コード内で特定の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 ;
21
KGs

上記の解決策は良いですが、さらに簡単にするために

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 ;
0
James