萬盛學電腦網

 萬盛學電腦網 >> 數據庫 >> mysql教程 >> mysql存儲過程中的異常處理

mysql存儲過程中的異常處理

MySQL 5.0開始地存儲過程,存儲過程最主要的還是執行效率和SQL 代碼封裝,增強了SQL語言的功能和靈活性,本文我們學習mysql存儲過程中的異常處理。

定義異常捕獲類型及處理方法:


    DECLARE handler_action HANDLER  
        FOR condition_value [, condition_value] ...  
        statement  
      
    handler_action:  
        CONTINUE  
      | EXIT  
      | UNDO  
      
    condition_value:  
        mysql_error_code  
      | SQLSTATE [VALUE] sqlstate_value  
      | condition_name  
      | SQLWARNING  
      | NOT FOUND  
      | SQLEXCEPTION  



這裡面需要注意幾點:

a、condition_value [,condition_value],這個的話說明可以包括多種情況(方括弧表示可選的),也就是一個handler可以定義成針對多種情況進行相應的 操作;另外condition_value可以包括的值有上面列出來的6種:

1、mysql_error_code,這個表示mysql的錯誤代碼,錯誤代碼是一個數字,完成是由mysql自己定義的,這個值可以參考mysql數據庫錯誤代碼及信息。
2、SQLSTATE [VALUE] sqlstate_value,這個同錯誤代碼類似形成一一對應的關系,它是一個5個字符組成的字符串,關鍵的地方是它從ANSI SQL和ODBC這些標准中引用過來的,因此更加標准化,而不像上面的error_code完全是mysql自己定義給自己用的,這個和第一個類似也可以 參考mysql數據庫錯誤代碼及信息。
3、condtion_name,這個是條件名稱,它使用DECLARE...CONDITION語句來定義,這個後面我們會介紹如何定義自己的condition_name。
4、SQLWARNING,表示SQLTATE中的字符串以‘01’起始的那些錯誤,比如Error: 1311 SQLSTATE: 01000 (ER_SP_UNINIT_VAR)
5、NOT FOUND,表示SQLTATE中的字符串以‘02’起始的那些錯誤,比如Error: 1329 SQLSTATE: 02000 (ER_SP_FETCH_NO_DATA)
6、SQLEXCEPTION,表示SQLSTATE中的字符串不是以'00'、'01'、'02' 起始的那些錯誤,這裡'00'起始的SQLSTATE其實表示的是成功執行而不是錯誤,另外兩個就是上面的4和5的兩種情況。

上面的6種情況其實可以分為兩類:

一類就是比較明確的處理,就是對指定的錯誤情況進行處理,包括1、2、3這三種方式;

另一類是對對應類型的錯誤的 處理,就是對某一群錯誤的處理,包括4、5、6這三種方式。這個是介紹了condition_value。另外還要注意的一個內容是MySQL在默認情況 下(也就是我們沒有定義處理錯誤的方法-handler)自己的錯誤處理機制:1、對於SQLWARNING和NOT FOUND的處理方法就是無視錯誤繼續執行,所以在游標的例子裡面如果我們沒有對repeat的條件判斷的那個值做個no_more_products=1的handler來處理,那麼循環就會一直下去。2、對於SQLEXCEPTION的話,其默認的處理方法是在出現錯誤的地方就終止掉了。

b、statement,這個比較簡單就是當出現某種條件/錯誤時,我們要執行的語句,可以是簡單的如 SET  var = value這樣的簡單的語句,也可以是復雜的多行的語句,多行的話可以使用BEGIN  .....  END這裡把語句包括在裡面(這個好比delphi裡面的情況,注意到我們的存儲過程也是多行的,所以也要BEGIN .... END)。
c、handler_action,這個表示當執行完上面的statement後,希望執行怎樣的動作,這裡包括CONTINUE、EXIT、UNDO, 表示繼續、退出、撤銷(暫時不支持)。這邊就是兩種動作,其實這兩種動作在上面也說過了,CONTINUE就是一個是SQLWARNING和NOT FOUND的默認處理方法,而EXIT就是SQLEXCEPTION的默認處理方法。



另:

condition_name:命名條件

MySQL error code或者SQLSTATE code的可讀性太差,所以引入了命名條件:

語法:

DECLARE condition_name CONDITION FOR condition_value  
 
condition_value:  
    SQLSTATE [VALUE] sqlstate_value  
  | mysql_error_code  

使用:

# original  
DECLARE CONTINUE HANDLER FOR 1216 MySQL_statements;  
 
# changed  
DECLARE foreign_key_error CONDITION FOR 1216;  
DECLARE CONTINUE HANDLER FOR foreign_key_error MySQL_statements;  


示例:

CREATE PROCEDURE sp_add_location  
    (in_location    VARCHAR(30),  
     in_address1    VARCHAR(30),  
     in_address2    VARCHAR(30),  
     zipcode        VARCHAR(10),  
     OUT out_status VARCHAR(30))  
BEGIN  
    DECLARE CONTINUE HANDLER  
        FOR 1062  
        SET out_status='Duplicate Entry';  
 
    SET out_status='OK';  
    INSERT INTO locations  
        (location,address1,address2,zipcode)  
    VALUES  
        (in_location,in_address1,in_address2,zipcode);  
END; 



MySQL 存儲過程 常用語法

存儲過程P_GET_CLASS_NAME是根據輸入的班級號判斷班級名稱

存儲過程P_INSERT_STUDENT是接收輸入的學生信息,最終將信息插入學生表。

    DROP PROCEDURE IF EXISTS `P_GET_CLASS_NAME`;  
    CREATE PROCEDURE P_GET_CLASS_NAME(IN ID int,OUT NAME VARCHAR(50))  
    BEGIN  
        IF(ID = 1) THEN   
              SET NAME = '一班';  
        END IF;  
        IF(ID = 2) THEN   
              SET NAME = '二班';  
        END IF;  
    END;  
      
      
    DROP PROCEDURE IF EXISTS `P_INSERT_STUDENT`;  
    CREATE PROCEDURE P_INSERT_STUDENT(IN ID INT,IN NAME VARCHAR(10),IN CLASSNO INT,IN BIRTH DATETIME)  
    BEGIN  
         SET @ID = ID;  
         SET @NAME = NAME;  
         SET @CLASSNO = CLASSNO;  
         SET @BIRTH = BIRTH;  
         SET @CLASSNAME = NULL;  
         CALL P_GET_CLASS_NAME(@CLASSNO,@CLASSNAME);  
           
         SET @insertSql = CONCAT('INSERT INTO TBL_STUDENT VALUES(?,?,?,?)');  
         PREPARE stmtinsert FROM @insertSql;  
         EXECUTE stmtinsert USING @ID,@NAME,@CLASSNAME,@BIRTH;  
         DEALLOCATE PREPARE stmtinsert;  
    END;  
      
    CALL P_INSERT_STUDENT(1,'xy',1,'2012-10-01 10:20:01');  

 

在第二個存儲過程中

①利用SET聲明了參數,調用了第一個存儲過程

②在第一個存儲過程中的NAME參數是輸出參數,所以@CLASSNAME這個參數在調用完第一個過程後就被附值

③最終利用CONCAT拼接SQL語句並傳入參數執行SQL語句


CALL P_INSERT_STUDENT(1,'xy',1,'2012-10-01 10:20:01');調用存儲過程

copyright © 萬盛學電腦網 all rights reserved