有时候,不希望存储过程抛出错误中止执行,而是希望返回一个错误码。 Mysql 支持异常处理,通过定义 CONTINUE/EXIT异常处理的 HANDLER 来捕获 SQLWARNING/NOT FOUND/SQLEXCEPTION (警告 / 无数据 / 其他异常)。其中, FOR 后面可以改为 SQLWARNING, NOT FOUND, SQLEXCEPTION 来指示所有异常都处理,相当于 oracle 中的 others 。例如,当不进行异常处理时,以下代码将直接抛出一个 ERROR 1062 (23000) 错误:
Sql code 4-1:
CREATE PROCEDURE test_proc_ins1(
IN i_id INT,
IN i_name VARCHAR(100)
)
BEGIN
INSERT INTO testproc VALUES (i_id,i_name);
INSERT INTO testproc VALUES (i_id,i_name);
END;
经过异常处理后,可以避免抛出错误,而是定义一个返回参数 o_ret 赋予特殊值来表示失败,这样,在 java 代码中,可以通过获取返回值而不是捕获异常的方式来处理业务逻辑。例如将返回值设置为 -1:
Sql code 4-2:
CREATE PROCEDURE test_proc_ins1(
IN i_id INT,
IN i_name VARCHAR(100),
OUT o_ret INT)
BEGIN
DECLARE EXIT HANDLER FOR SQLSTATE '23000' set o_ret = -1;
-- 也可以这样使用:
-- DECLARE EXIT HANDLER FOR SQLWARNING,NOT FOUND,SQLEXCEPTION set o_ret=-1;
INSERT INTO testproc VALUES (i_id,i_name);
INSERT INTO testproc VALUES (i_id,i_name);
set o_ret = 1;
调用:call
test_proc_ins1(1,"tom",@outret);
select @outret;
DECLARE HANDLER syntax
声明异常处理的语法
DECLARE
{ EXIT | CONTINUE }
HANDLER FOR
{ error-number | { SQLSTATE error-string } | condition }
SQL statement
上面就是错误处理的用法,也就是一段当程序出错后自动触发的代码。
MySQL
允许两种
处理,
一种是 EXIT 处理,
另一种是
CONTINUE
处理,它跟 EXIT 处理类似,不同在于它执行后,原主程序仍然继续运行,那么这个复合语
句就没有出
口了。
1. DECLARE CONTINUE HAND
LER example
CREATE TABLE t4 (s1 int,primary key(s1));
CREATE PROCEDURE p23 ()
BEGIN
DECLARE CONTINUE HANDLER FOR SQLSTATE '23000' SET @x2 = 1;
SET @x = 1;
INSERT INTO t4 VALUES (1);
SET @x = 2;
INSERT INTO t4 VALUES (1);
SET @x = 3;
END;
这是
MySQL
参考手册上的 CONTINUE 处理的例子,这个例子十分好,
通过这个例子我们可以看出 CONTINUE 处理是如何工作的。
2. DECLARE CONTINUE HANDLER 声明 CONTINUE 异常处理
这次我将为 SQLSTATE 值定义一个处理程序。还记得前面我们使用的
MySQL
错误代码
1216 吗?
事实上这里的 23000SQLSTATE 是更常用的,当外键约束出错或主键约束出错就被调用
了。
1、这个
存储过程
的第一个执行的语句是"SET @x = 1"。
2、运行后值 1 被插入到主键表中。
3、然后@x 的值变为 2。
4、然后程序尝试再次往主键表中插入数值,但失败了,因为主键有唯一性限制。
5、由于插入失败,错误处理程序被触发,开始进行错误处理。下一个执行的语句是错误处
理的语句,@x2 被设为1。
6、到这里没有结束,因为这是 CONTINUE 异常处理。所以执行返回到失败的插入语句
后,继续执行将@x 设定为 3 动作。
DECLARE CONTINUE HANDLER example
mysql> CALL p23()
Query OK, 0 rows affected (0.00 sec)
mysql> SELECT @x, @x2
+------+------+
| @x | @x2 |
+------+------+
| 3 | 1 |
+------+------+
1 row in set (0.00 sec)
运行过程后,我们观察@x 的值,很确定的可以知道是 3,观察@x2 的值,为 1。从这里
可以判断程序运行无误,完全按照我们的思路进行。大家可以花点时间去调整错误处理器,
让检查放在语句段的首部,而不是放在可能出现错误的地方,虽然那样看起来程序很紊乱,
跳来跳去的感觉。但是这样的代码很安全也很清楚。
DECLARE CONDITION
CREATE PROCEDURE p24 ()
BEGIN
DECLARE `Constraint Violation` CONDITION FOR SQLSTATE '23000';
DECLARE EXIT HANDLER FOR
`Constraint Violation` ROLLBACK;
START TRANSACTION;
INSERT INTO t2 VALUES (1);
INSERT INTO t2 VALUES (1);
COMMIT;
END;
这是另外一个错误处理的例子,在前面的基础上修改的。事实上你可给 SQLSTATE 或者
错误代码起名字,你就可以在处理中使用自己定义的名字了。下面看它是怎么实现的:
我把表 t2 定义为 InnoDB 表,所以对这个表的插入操作都会 RO
LLBACK(回滚),ROLLBACK
也是恰好会发生的。因为对主键插入两个同样的值会导致 SQLSTATE 23000 错
误发生,这里 SQLSTATE 23000 是约束错误。
mysql> CALL p24()
Query OK, 0 rows affected (0.28 sec)
mysql> SELECT * FROM t2//
Empty set (0.00 sec)
调用这个
存储过程
看结果是什么,从上面结果我们看到表 t2 没有插入任何记录。
全部事务都回滚了。这正是我们想要的。