Mysqlコマンドのリストでエラーが発生した場合、自動的にロールバックできますか?
たとえば、次のような行に沿ったものです。
begin transaction;
insert into myTable values1 ...
insert into myTable values2 ...; -- will throw an error
commit;
今、実行時にトランザクション全体が失敗するようにしたいので、[〜#〜] not [〜#〜] myTableのvalues1を参照してください。しかし、残念ながら、トランザクションにエラーがある場合でも、テーブルにはvalues1が挿入されます。
ロールバックする方法はありますか? (再度、エラーが発生した場合)?
編集-DDLから標準SQLに変更
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 を確認してください。
たとえば、コード内で特定の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 ;
上記の解決策は良いですが、さらに簡単にするために
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 ;