MySQL 存储过程错误处理

Posted

技术标签:

【中文标题】MySQL 存储过程错误处理【英文标题】:MySQL Stored Procedure Error Handling 【发布时间】:2011-12-07 13:42:12 【问题描述】:

我相信目前 mysql 中没有任何可用的东西允许访问 MySQL 存储过程中最后执行的语句的SQLSTATE。这意味着当在存储过程中引发泛型 SQLException 时,很难/不可能得出错误的确切性质。

是否有人有解决方法来导出 MySQL 存储过程中的错误SQLSTATE,而不涉及为每个可能的 SQLSTATE 声明处理程序?

例如 - 假设我试图返回一个 error_status,它超出了以下通用的“SQLException 在此BEGIN....END 块中的某处发生”:

DELIMITER $$

CREATE PROCEDURE `myProcedure`(OUT o_error_status varchar(50))
MY_BLOCK: BEGIN

 DECLARE EXIT handler for 1062 set o_error_status := "Duplicate entry in table";
 DECLARE EXIT handler for 1048 set o_error_status := "Trying to populate a non-null column with null value"; 
-- declare handlers ad nauseum here....

 DECLARE EXIT handler for sqlexception set o_error_status:= "Generic SQLException. You'll just have to figure out the SQLSTATE yourself...." ;

-- Procedure logic that might error to follow here...

END MY_BLOCK$$

有什么建议吗?

PS 我正在运行 MySQL 5.1.49

【问题讨论】:

【参考方案1】:

GET DIAGNOSTICS 在 5.6.4 中可用

看 http://dev.mysql.com/doc/refman/5.6/en/get-diagnostics.html

【讨论】:

谢谢!真的很有用。我现在要做的就是升级到 5.6 ;-)【参考方案2】:

我正在执行以下解决方法:使用 SELECT 来引发错误。例如:

SELECT RAISE_ERROR_unable_to_update_basket;

这将导致以下错误消息(示例):

ERROR 1054 (42S22): Unknown column 'RAISE_ERROR_unable_to_update_basket' in 'field list'

我将对存储过程的调用包装在 try ... catch ... 中,现在可以处理此错误。当然,这仅适用于从存储过程内部引发自定义错误消息,并且不会处理可能发生的任何 SQL 或数据库错误(由于重复键条目)。在后一种情况下,您也许可以使用 Johan 的解决方案来解决此问题。

【讨论】:

谢谢哈拉德。虽然在这个特定的实例中,我正在尝试处理 SQL 或数据库错误。【参考方案3】:

我相信目前 MySQL 中没有任何东西可以访问 MySQL 存储过程中最后执行的语句的 SQLSTATE。这意味着......很难/不可能得出错误的确切性质。

幸运的是,这不是真的。

SHOW ERRORS LIMIT 1   -- for SQL-state > 2
SHOW WARNINGS LIMIT 1 -- for SQL-state 1,2

将显示最后的错误或警告。

为了防止列出每个错误,您可以像这样处理一类 SQL 错误:

SQLWARNING 是以 '01' 开头的 SQLSTATE 值类的简写。

NOT FOUND 是以 '02' 开头的 SQLSTATE 值类的简写。这仅在游标的上下文中相关,用于控制当游标到达数据集末尾时发生的情况。如果没有更多行可用,则会出现 SQLSTATE 值为 02000 的 No Data 条件。要检测此条件,您可以为其设置处理程序(或为 NOT FOUND 条件)。第 12.7.5 节“游标”中显示了一个示例。 SELECT ... INTO var_list 不检索任何行的语句也会出现这种情况。

SQLEXCEPTION 是不以“00”、“01”或“02”开头的 SQLSTATE 值类的简写。

所以要处理异常,您需要做:

DECLARE EXIT HANDLER FOR SQLSTATE SQLEXCEPTION .....;

链接:http://dev.mysql.com/doc/refman/5.5/en/signal.htmlhttp://dev.mysql.com/doc/refman/5.0/en/declare-handler.html

【讨论】:

谢谢约翰。如果单独运行,SHOW ERRORSSHOW WARNINGS 确实会显示最后一个错误或警告的 SQLSTATE。但是,我如何访问“在 MySQL 存储过程中”?也就是说,如何在我原来的问题中描述的过程输出变量中返回SHOW ERRORSSHOW WARNINGS 的输出? 如何将SHOW ERRORS 文本的结果存储在变量中?

以上是关于MySQL 存储过程错误处理的主要内容,如果未能解决你的问题,请参考以下文章

MySQL存储过程错误处理

Mysql存储过程——异常处理

MySQL存储过程的“异常处理”

mysql 存储过程提示错误!!求高人解答

MySQL数据库 *实验18错误处理

关于MySQL存储过程异常处理的一点心得