1. 程式人生 > >如何在mysql下實現事務的提交與回滾(try&catch)

如何在mysql下實現事務的提交與回滾(try&catch)

最近要對資料庫的資料進行一個定時遷移,為了防止在執行過程sql語句因為某些原因報錯而導致資料轉移混亂,因此要對我們的指令碼加以事務進行控制。

首先我們建一張tran_test表

CREATE TABLE tran_test(  
f1 VARCHAR(10) NOT NULL,  
f2 INT(1) DEFAULT NULL,  
PRIMARY KEY (f1)  
)ENGINE=INNODB CHARSET=utf8

我想對tran_test插入兩條資料,但是為了防止插入中報錯,因此我要把插入語句控制在一個事務內。

這時候,如果你查一下有些人的文章,許多時候會給出你這麼一條答案。

START TRANSACTION;    
INSERT INTO tran_test VALUES('A',1);  
INSERT INTO tran_test VALUES('B',2);  
ROLLBACK;

START TRANSACTION;    
INSERT INTO tran_test VALUES('A',1);  
INSERT INTO tran_test VALUES('B',2);  
COMMIT;

看上去很簡單的sql語句,並且這兩句也確實能實現提交或回滾。

 

然而這真的能達到我們的目的嗎?答案是否定的。

比如第一段,它是將你在事務中的sql語句無論對錯全部進行ROLLBACK。這樣絕對的回滾使得你的sql沒有任何意義了。

 

因此我們想要真正的控制好事務,我的思路是對要執行的sql進行異常檢測。如果sql沒有出現異常,COMMIT,如果捕獲到了異常,則ROLLBACK。

這時候,我們就需要建一個儲存過程來捕獲異常。執行成功時進行COMMIT,sql執行失敗時則進行ROLLBACK。

兩種思路可以達到我想要的效果。

 

第一種是對我們要執行的sql進行異常捕獲,我們再定義一個變數t_error,當捕獲到異常的時候,讓t_error=1。再對t_error進行條件判斷,如果t_error=1則進行ROLLBACK,否則進行COMMIT。

DROP PROCEDURE IF EXISTS t_test;  
DELIMITER //  
CREATE PROCEDURE t_test()  
  BEGIN  
    DECLARE t_error INTEGER;  
    DECLARE CONTINUE HANDLER FOR SQLEXCEPTION SET t_error = 1;  
    START TRANSACTION;  
         INSERT INTO tran_test VALUES('A',1);  
     INSERT INTO tran_test VALUES('B',2);  
         IF t_error = 1 THEN  
             ROLLBACK;  
         ELSE  
             COMMIT;  
         END IF;  
END//  
CALL t_test();

另一隻則是第一種的簡化,即捕獲到異常直接進行ROLLBACK,如果沒捕獲到異常,直接COMMIT

DROP PROCEDURE  IF EXISTS t_test;  
DELIMITER //  
CREATE PROCEDURE t_test()  
BEGIN  
DECLARE EXIT HANDLER FOR SQLEXCEPTION ROLLBACK;  
START TRANSACTION;  
INSERT INTO tran_test VALUES('A',1);  
INSERT INTO tran_test VALUES('B',2);  
COMMIT;  
END//  
CALL t_test()

這樣,這兩個insert語句便真正的被控制在了