mysql存储过程之错误处理实例详解
本文实例讲述了mysql存储过程之错误处理。分享给大家供大家参考,具体如下:
当存储过程中发生错误时,重要的是适当处理它,例如:继续或退出当前代码块的执行,并发出有意义的错误消息。其中mysql提供了一种简单的方法来定义处理从一般条件(如警告或异常)到特定条件(例如特定错误代码)的处理程序。完事我们来使用DECLAREHANDLER语句来尝试声明一个处理程序,先来看语法:
DECLAREactionHANDLERFORcondition_valuestatement;
上述sql中,如果条件的值与condition_value匹配,则MySQL将执行statement,并根据该操作继续或退出当前的代码块。其中,操作(action)接受以下值之一:
- CONTINUE:继续执行封闭代码块(BEGIN...END)。
- EXIT:处理程序声明封闭代码块的执行终止。
condition_value指定一个特定条件或一类激活处理程序的条件。condition_value接受以下值之一:
- 一个MySQL错误代码。
- 标准SQLSTATE值或者它可以是SQLWARNING,NOTFOUND或SQLEXCEPTION条件,这是SQLSTATE值类的简写。NOTFOUND条件用于游标或SELECTINTOvariable_list语句。
- 与MySQL错误代码或SQLSTATE值相关联的命名条件。
最重要的是,上述sql可以是一个简单的语句或由BEGIN和END关键字包围的复合语句。介绍完事之后,咱们来看几个声明处理程序的例子,首先是当程序发生错误时,将has_error变量的值设置为1并继续执行的例子:
DECLARECONTINUEHANDLERFORSQLEXCEPTIONSEThas_error=1;
再来看当发生错误时,回滚上一个操作,发出错误消息,并退出当前代码块。如果在存储过程的BEGINEND块中声明它,则会立即终止存储过程:
DECLAREEXITHANDLERFORSQLEXCEPTION BEGIN ROLLBACK; SELECT'Anerrorhasoccurred,operationrollbackedandthestoredprocedurewasterminated'; END;
以下处理程序的意思是,如果没有更多的行要提取,在光标或selectinto语句的情况下,将no_row_found变量的值设置为1并继续执行:
DECLARECONTINUEHANDLERFORNOTFOUNDSETno_row_found=1;
以下处理程序如果发生重复的键错误,则会发出MySQL错误1062。它发出错误消息并继续执行:
DECLARECONTINUEHANDLERFOR1062 SELECT'Error,duplicatekeyoccurred';
上面这些实例可能有点抽象,咱们废话不多说,先来创建一个名为article_tags的新表,来具体操作下:
USEtestdb; CREATETABLEarticle_tags( article_idINT, tag_idINT, PRIMARYKEY(article_id,tag_id) );
其中呢,article_tags表存储文章和标签之间的关系。每篇文章可能有很多标签,反之亦然。为了简单起见,我们不会在article_tags表中创建文章(article)表和标签(tags)表以及外键。
完事呢,我们来创建一个存储过程,将文章的id和标签的id插入到article_tags表中:
USEtestdb; DELIMITER$$ CREATEPROCEDUREinsert_article_tags(INarticle_idINT,INtag_idINT) BEGIN DECLARECONTINUEHANDLERFOR1062 SELECTCONCAT('duplicatekeys(',article_id,',',tag_id,')found')ASmsg; --insertanewrecordintoarticle_tags INSERTINTOarticle_tags(article_id,tag_id) VALUES(article_id,tag_id); --returntagcountforthearticle SELECTCOUNT(*)FROMarticle_tags; END$$ DELIMITER;
然后呢,我们通过调用insert_article_tags存储过程,为文章ID为1添加标签ID:1,2和3,如下所示:
CALLinsert_article_tags(1,1); CALLinsert_article_tags(1,2); CALLinsert_article_tags(1,3);
我们再尝试插入一个重复的键来检查处理程序是否真的被调用:
CALLinsert_article_tags(1,3);
执行上面查询语句,得到以下结果:
mysql>CALLinsert_article_tags(1,3); +----------------------------+ |msg| +----------------------------+ |duplicatekeys(1,3)found| +----------------------------+ 1rowinset +----------+ |COUNT(*)| +----------+ |3| +----------+ 1rowinset QueryOK,0rowsaffected
执行后会收到一条错误消息。但是,由于我们将处理程序声明为CONTINUE处理程序,所以存储过程继续执行。因此,最后获得了文章的标签计数值为:3。来看个图:
但是如果将处理程序声明中的CONTINUE更改为EXIT,那么将只会收到一条错误消息。如下查询语句:
DELIMITER$$ CREATEPROCEDUREinsert_article_tags_exit(INarticle_idINT,INtag_idINT) BEGIN DECLAREEXITHANDLERFORSQLEXCEPTION SELECT'SQLExceptioninvoked'; DECLAREEXITHANDLERFOR1062 SELECT'MySQLerrorcode1062invoked'; DECLAREEXITHANDLERFORSQLSTATE'23000' SELECT'SQLSTATE23000invoked'; --insertanewrecordintoarticle_tags INSERTINTOarticle_tags(article_id,tag_id) VALUES(article_id,tag_id); --returntagcountforthearticle SELECTCOUNT(*)FROMarticle_tags; END$$ DELIMITER;
执行上面查询语句,得到以下结果:
mysql>CALLinsert_article_tags_exit(1,3); +-------------------------------+ |MySQLerrorcode1062invoked| +-------------------------------+ |MySQLerrorcode1062invoked| +-------------------------------+ 1rowinset QueryOK,0rowsaffected
来看个图感受下:
如果我们使用多个处理程序来处理错误,MySQL将调用最特定的处理程序来处理错误。这就涉及到优先级的问题了,我们来具体看下。
我们知道错误总是映射到一个MySQL错误代码,因为在MySQL中它是最具体的。SQLSTATE可以映射到许多MySQL错误代码,因此它不太具体。SQLEXCPETION或SQLWARNING是SQLSTATES类型值的缩写,因此它是最通用的。假设在insert_article_tags_3存储过程中声明三个处理程序,如下所示:
DELIMITER$$ CREATEPROCEDUREinsert_article_tags_3(INarticle_idINT,INtag_idINT) BEGIN DECLAREEXITHANDLERFOR1062SELECT'Duplicatekeyserrorencountered'; DECLAREEXITHANDLERFORSQLEXCEPTIONSELECT'SQLExceptionencountered'; DECLAREEXITHANDLERFORSQLSTATE'23000'SELECT'SQLSTATE23000'; --insertanewrecordintoarticle_tags INSERTINTOarticle_tags(article_id,tag_id) VALUES(article_id,tag_id); --returntagcountforthearticle SELECTCOUNT(*)FROMarticle_tags; END$$ DELIMITER;
然后我们尝试通过调用存储过程将重复的键插入到article_tags表中:
CALLinsert_article_tags_3(1,3);
如下,可以看到MySQL错误代码处理程序被调用:
mysql>CALLinsert_article_tags_3(1,3); +----------------------------------+ |Duplicatekeyserrorencountered| +----------------------------------+ |Duplicatekeyserrorencountered| +----------------------------------+ 1rowinset QueryOK,0rowsaffected
完事之后,咱们再来看下使用命名错误条件。从错误处理程序声明开始,如下:
DECLAREEXITHANDLERFOR1051SELECT'Pleasecreatetableabcfirst'; SELECT*FROMabc;
1051号是什么意思?想象一下,你有一个大的存储过程代码使用了好多类似这样的数字;这将成为维护代码的噩梦。幸运的是,MySQL为我们提供了声明一个命名错误条件的DECLARECONDITION语句,它与条件相关联。DECLARECONDITION语句的语法如下:
DECLAREcondition_nameCONDITIONFORcondition_value;
condition_value可以是MySQL错误代码,例如:1015或SQLSTATE值。condition_value由condition_name表示。声明后,可以参考condition_name,而不是参考condition_value。所以可以重写上面的代码如下:
DECLAREtable_not_foundCONDITIONfor1051; DECLAREEXITHANDLERFORtable_not_foundSELECT'Pleasecreatetableabcfirst'; SELECT*FROMabc;
这段代码比以前的代码显然更可读,不过我们要注意,条件声明必须出现在处理程序或游标声明之前。
好啦,这次就到这里了。
更多关于MySQL相关内容感兴趣的读者可查看本站专题:《MySQL存储过程技巧大全》、《MySQL常用函数大汇总》、《MySQL日志操作技巧大全》、《MySQL事务操作技巧汇总》及《MySQL数据库锁相关技巧汇总》
希望本文所述对大家MySQL数据库计有所帮助。
声明:本文内容来源于网络,版权归原作者所有,内容由互联网用户自发贡献自行上传,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任。如果您发现有涉嫌版权的内容,欢迎发送邮件至:czq8825#qq.com(发邮件时,请将#更换为@)进行举报,并提供相关证据,一经查实,本站将立刻删除涉嫌侵权内容。