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小提琴.
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)
上面的解决方案很好,但让它变得更简单
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)