最近要對數據庫的數據進行一個定時遷移,為了防止在執行過程sql語句因為某些原因報錯而導致數據轉移混亂,因此要對我們的腳本加以事務進行控制。
首先我們建一張tran_test表
1
2
3
4
5
|
CREATE TABLE tran_test( f1 VARCHAR (10) NOT NULL , f2 INT (1) DEFAULT NULL , PRIMARY KEY (f1) )ENGINE=INNODB CHARSET=utf8 |
我想對tran_test插入兩條數據,但是為了防止插入中報錯,因此我要把插入語句控制在一個事務內。
這時候,如果你查一下有些人的文章,許多時候會給出你這么一條答案。
1
2
3
4
|
START TRANSACTION ; INSERT INTO tran_test VALUES ( 'A' ,1); INSERT INTO tran_test VALUES ( 'B' ,2); ROLLBACK ; |
或
1
2
3
4
|
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。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
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
1
2
3
4
5
6
7
8
9
10
11
|
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語句便真正的被控制在了一個事務內了。
以上實例大家可以在本次測試一下,如果有其他補充和疑問可以直接聯系小編,感謝大家對服務器之家的支持。
原文鏈接:https://www.cnblogs.com/lykbk/p/sdfdfdf23423434345.html