MySQL定义异常和异常处理方法

在MySQL中,特定异常需要特定处理。这些异常可以联系到错误,以及子程序中的一般流程控制。定义异常是事先定义程序执行过程中遇到的问题,异常处理定义了在遇到问题时对应当采取的处理方式,并且保证存储过程或者函数在遇到错误时或者警告时能够继续执行。


1 异常定义

1.1 语法

DECLARE condition_name CONDITION FOR [condition_type];

1.2 说明

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为数值类型错误代码;

1.3 示例

定义“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;


2 自定义异常处理

2.1 异常处理语法

DECLARE handler_type HANDLER FOR condition_value [,...] sp_statement

2.2 参数说明

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匹配数值类型错误代码;


2.3 异常捕获方法

//方法一:捕获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 综合示例

创建一个表,设置该表的主键,在不定义异常处理和定义异常处理情况下看执行到哪一步。

show databases;
use wms;
create table location
(
location_id int primary key,
location_name varchar(50)
);

示例1:不定义异常情况下

DELIMITER //
CREATE PROCEDURE handlerInsertNoException()
BEGIN
	/*DECLARE CONTINUE HANDLER FOR SQLSTATE '23000' SET @x2=1;*/
	SET @x=1;
	INSERT INTO location VALUES (1,'Beijing');
	SET @x=2;
	INSERT INTO location VALUES (1,'Wuxi');
	SET @x=3;
END;
//
DELIMITER ;
调用存储过程与结果:

mysql> call handlerInsertNoException();
ERROR 1062 (23000): Duplicate entry '1' for key 'PRIMARY'
mysql> select @x;
+------+
| @x   |
+------+
|    2 |
+------+
1 row in set (0.00 sec)

mysql> select * from location;
+-------------+---------------+
| location_id | location_name |
+-------------+---------------+
|           1 | Beijing       |
+-------------+---------------+
1 row in set (0.00 sec)


 

注意:操作示例2前要清空表中数据,并退出重新登录,以免客户端变量@x影响,详细说明参见结论中的第一点。

mysql> truncate table location;
Query OK, 0 rows affected (0.04 sec)
mysql> select * from location;
Empty set (0.00 sec)
mysql> exit;
Bye

david@Louis:~$ mysql -u root -p
Enter password: 
Welcome to the MySQL monitor.  Commands end with ; or \g.
Your MySQL connection id is 53
Server version: 5.5.38-0ubuntu0.14.04.1 (Ubuntu)

mysql> use wms;
Reading table information for completion of table and column names
You can turn off this feature to get a quicker startup with -A

Database changed
mysql> select * from location;
Empty set (0.00 sec)

mysql> select @x;
+------+
| @x   |
+------+
| NULL |
+------+
1 row in set (0.00 sec)


示例2:定义异常处理情况下:

DELIMITER //
CREATE PROCEDURE handlerInsertWithException()
BEGIN
	DECLARE CONTINUE HANDLER FOR SQLSTATE '23000' SET @x2=1;
	SET @x=1;
	INSERT INTO location VALUES (1,'Beijing');
	SET @x=2;
	INSERT INTO location VALUES (1,'Wuxi');
	SET @x=3;
END;
//
DELIMITER ;

调用存储过程与结果:

mysql> CALL handlerInsertWithException();
Query OK, 0 rows affected (0.09 sec)

mysql> select @x;
+------+
| @x   |
+------+
|    3 |
+------+
1 row in set (0.00 sec)

说明与结论:

一、MySQL中,@var_name表示用户变量,使用SET语句为其赋值,用户变量与连接有关,一个客户端定义的变量不能被其他客户端看到或者使用。当客户端退出时,该客户端连接的所有变量将自动释放。

二、在示例1中,由于注释了异常的声明"",此时向表中插入相同主键,就会触发异常,并且采取默认(EXIT)路径;且查看此时的@x返回2,表示下面的INSERT语句并没有执行就退出了.

三、定义了异常处理,此时遇到错误也会按照异常定义那样继续执行;但只有第一条数据被插入到表中,此时用户变量@x=3说明已经执行到了结尾;


-----------------------------------------------------------------------------------------------------------------------------

如果您们在尝试的过程中遇到什么问题或者我的代码有错误的地方,请给予指正,非常感谢!

联系方式:david.louis.tian@outlook.com

版权@:转载请标明出处!
  • 6
    点赞
  • 17
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
MySQL 中,可以使用异常处理来处理存储过程中的异常情况。MySQL 提供了两种类型的异常条件异常和 SQLSTATE 异常条件异常是基于条件的自定义异常,在存储过程中使用 SIGNAL 语句来抛出异常。下面是一个示例: ```sql CREATE PROCEDURE example_procedure() BEGIN DECLARE custom_error CONDITION FOR SQLSTATE '45000'; DECLARE EXIT HANDLER FOR custom_error BEGIN -- 异常处理逻辑 -- 可以使用 GET DIAGNOSTICS 语句获取异常信息 GET DIAGNOSTICS CONDITION 1 @sql_state = RETURNED_SQLSTATE, @errno = MYSQL_ERRNO, @text = MESSAGE_TEXT; -- 输出异常信息 SELECT CONCAT('Error: ', @text) AS ErrorMessage; END; -- 抛出异常 SIGNAL custom_error SET MESSAGE_TEXT = 'Custom error message'; END; ``` 在上面的示例中,我们创建了一个名为 `example_procedure` 的存储过程。在存储过程中,我们定义了一个名为 `custom_error` 的条件异常。然后,我们使用 `DECLARE EXIT HANDLER FOR custom_error` 语句来指定当 `custom_error` 异常被抛出时要执行的处理程序。在处理程序中,我们使用 `GET DIAGNOSTICS` 语句获取异常信息,并输出异常信息。 另一种类型的异常是 SQLSTATE 异常,它是基于 MySQL 错误代码的系统定义异常。当出现预定义的错误代码时,MySQL 会自动抛出异常。可以使用 `DECLARE ... HANDLER FOR SQLEXCEPTION` 或 `DECLARE ... HANDLER FOR SQLWARNING` 语句来指定异常处理程序。 下面是一个使用 SQLSTATE 异常处理的示例: ```sql CREATE PROCEDURE example_procedure() BEGIN DECLARE EXIT HANDLER FOR SQLEXCEPTION BEGIN -- 异常处理逻辑 SELECT CONCAT('Error: ', SQLSTATE) AS ErrorMessage; END; -- 引发异常 SELECT 1 / 0; END; ``` 在这个例子中,我们创建了一个名为 `example_procedure` 的存储过程。我们使用 `DECLARE EXIT HANDLER FOR SQLEXCEPTION` 语句指定当发生任何异常时要执行的处理程序。在处理程序中,我们使用 `SQLSTATE` 函数获取异常的 SQLSTATE 值,并输出异常信息。 这些都是处理 MySQL 存储过程中异常情况的一些基本方法。你可以根据实际需求来选择适当的处理方式和异常类型。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值