--該文章內容通過網絡搜索組合,
mysql 異常,可以自定義異常,再應用。也可使用系統默認的異常,捕獲應用。
一、異常定義:
DECLARE condition_name CONDITION FOR [condition_type];
condition_name參數表示異常的名稱;
condition_type參數異常類型
condition_type由SQLSTATE [VALUE] sqlstate_value|mysql_error_code組成:
sqlstate_value和mysql_error_code都可以表示MySQL的錯誤;
sqlstate_value為長度為5的字符串類型的錯誤代碼;mysql_error_code為數值類型錯誤代碼;
舉例:
定義“ERROR 1148(42000)”錯誤,名稱為command_not_allowed。可以有以下兩種方法:
//方法一:使用sqlstate_value
DECLARE command_not_allowed CONDITION FOR SQLSTATE '42000';
//方法二:使用mysql_error_code
DECLARE command_not_allowed CONDITION FOR 1148;
二、異常處理
DECLARE handler_type HANDLER FOR condition_value [,...] sp_statement
handler_type: CONTINUE|EXIT|UNDO
handler_type為錯誤處理方式,參數為3個值之一;
CONTINUE表示遇到錯誤不處理,繼續執行;
EXIT表示遇到錯誤時馬上退出;
UNDO表示遇到錯誤后撤回之前的操作,MySQL暫不支持回滾操作;
condition_value: SQLSTATE [VALUE] sqlstate_value| condition_name|SQLWARNING|NOT FOUND|SQLEXCEPTION|mysql_error_code
condition_value表示錯誤類型;SQLSTATE [VALUE] sqlstate_value為包含5個字符的字符串錯誤值;
condition_name表示DECLARE CONDITION定義的錯誤條件名稱;
SQLWARNING匹配所有以01開頭的SQLSTATE錯誤代碼;
NOT FOUND匹配所有以02開頭的SQLSTATE錯誤代碼;
SQLEXCEPTION匹配所有沒有被SQLWARNING或NOT FOUND捕獲的SQLSTATE錯誤代碼;
mysql_error_code匹配數值類型錯誤代碼;
舉例:
//方法一:捕獲sqlstate_value異常
//這種方法是捕獲sqlstate_value值。如果遇到sqlstate_value值為"42S02",執行CONTINUE操作,並輸出"NO_SUCH_TABLE"信息
DECLARE CONTINUE HANDLER FOR SQLSTATE '42S02' SET @info='NO_SUCH_TABLE';
//方法二:捕獲mysql_error_code異常
//這種方法是捕獲mysql_error_code值。如果遇到mysql_error_code值為1146,執行CONTINUE操作,並輸出"NO_SUCH_TABLE"信息;
DECLARE CONTINUE HANDLER FOR 1146 SET @info='NO_SUCH_TABLE';
//方法三:先定義條件,然后捕獲異常
DECLARE no_such_table CONDITION FOR 1146;
DECLARE CONTINUE HANDLER FOR NO_SUCH_TABLE SET @info='NO_SUCH_TABLE';
//方法四:使用SQLWARNING捕獲異常
DECLARE EXIT HANDLER FOR SQLWARNING SET @info='ERROR';
//方法五:使用NOT FOUND捕獲異常
DECLARE EXIT HANDLER FOR NOT FOUND SET @info='NO_SUCH_TABLE';
//方法六:使用SQLEXCEPTION捕獲異常
DECLARE EXIT HANDLER FOR SQLEXCEPTION SET @info='ERROR';
3.實例
1.
CREATE PROCEDURE P_TEST_EXCEP()
BEGIN
DECLARE EXIT HANDLER for SQLEXCEPTION INSERT into TEST_LOG(id) VALUES(1);
START TRANSACTION;
INSERT into TEST(id) VALUES(11);--主鍵不沖突
INSERT into TEST(id) VALUES(1);--主鍵沖突
INSERT into TEST(id) VALUES(111);--主鍵不沖突
commit;
END
--實際發現,此時日志表也被回滾了。
2、
CREATE PROCEDURE P_TEST_EXCEP()
BEGIN
declare t_error int default 0;
DECLARE CONTINUE HANDLER for SQLEXCEPTION SET t_error=1;
START TRANSACTION;
INSERT into TEST(id) VALUES(11);--主鍵不沖突
INSERT into TEST(id) VALUES(1);--主鍵沖突
INSERT into TEST(id) VALUES(111);--主鍵不沖突
if t_error=1 THEN
ROLLBACK;
ELSE
commit;
end if;
END
3.
CREATE PROCEDURE P_TEST_EXCEP()
BEGIN
DECLARE EXIT HANDLER for SQLEXCEPTION begin ROLLBACK ; START TRANSACTION;INSERT into TEST_LOG(id) VALUES(1); commit; end;
START TRANSACTION;
INSERT into TEST(id) VALUES(11);--主鍵不沖突
INSERT into TEST(id) VALUES(1);--主鍵沖突
INSERT into TEST(id) VALUES(111);--主鍵不沖突
commit;
END
--在異常處,先回滾之前開啟的事物,再重新開啟事物,提交日志信息。
