问题 MySQL存储过程错误处理


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

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

例如 - 想象一下,我试图返回一个超出泛型的错误_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


7739
2017-10-14 08:19


起源



答案:


GET DIAGNOSTICS可在5.6.4中找到

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


7
2018-01-19 09:10



谢谢!真有用。我现在要做的就是升级到5.6 ;-) - Tom Mac


答案:


GET DIAGNOSTICS可在5.6.4中找到

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


7
2018-01-19 09:10



谢谢!真有用。我现在要做的就是升级到5.6 ;-) - Tom Mac


我相信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时出现无数据条件。要检测此情况,可以为其设置处理程序(或者对于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.html
http://dev.mysql.com/doc/refman/5.0/en/declare-handler.html 


9
2017-10-14 09:54



谢谢约翰。该 SHOW ERRORS 和 SHOW WARNINGS 如果单独运行,确实会向我显示上一个错误或警告的SQLSTATE。但是,如何在“MySQL存储过程中”访问它?也就是说,我怎样才能返回输出 SHOW ERRORS要么 SHOW WARNINGS 在我原始问题中描述的过程输出变量中? - Tom Mac
如何存储结果 SHOW ERRORS  变量中的文字? - Ahmed


我正在做以下的解决方法:使用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的解决方案解决此问题。


0
2017-10-14 09:56



谢谢哈拉尔德。虽然在这个特定的实例中我试图处理SQL或数据库错误。 - Tom Mac