存储过程(四)异常处理

有时候,不希望存储过程抛出错误中止执行,而是希望返回一个错误码。 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;  
     END;  

调用: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 没有插入任何记录。 全部事务都回滚了。这正是我们想要的。 

你可能感兴趣的:(sql,异常处理,存储过程,continue,exit)