`
wwty
  • 浏览: 544295 次
  • 性别: Icon_minigender_1
  • 来自: 北京-郑州
社区版块
存档分类
最新评论

mysql存储过程之异常处理篇

阅读更多

mysql存储过程也提供了对异常处理的功能:通过定义HANDLER来完成异常声明的实现

 

 

语法如下:

DECLARE handler_type HANDLER FOR condition_value[,...] sp_statement handler_type: CONTINUE | EXIT condition_value: SQLSTATE [VALUE] sqlstate_value | condition_name | SQLWARNING | NOT FOUND | SQLEXCEPTION | mysql_error_code


Handlers类型:

1, EXIT: 发生错误时退出当前代码块(可能是子代码块或者main代码块)
2, CONTINUE: 发送错误时继续执行后续代码

condition_value:

condition_value支持标准的SQLSTATE定义;

SQLWARNING是对所有以01开头的SQLSTATE代码的速记

NOT FOUND是对所有以02开头的SQLSTATE代码的速记

SQLEXCEPTION是对所有没有被SQLWARNINGNOT FOUND捕获的SQLSTATE代码的速记

除了SQLSTATE值,MySQL错误代码也被支持

但是对于mysql而言,优先级如下:
MySQL Error code > SQLSTATE code > 命名条件

 

 

使用SQLSTATE还是MySQL Error Code? 

1,SALSTATE是标准,貌似会更portable,但是实际上MySQL、DB2、Oracle等等的存储程序语法大相径庭,所以portable的优势不存在 
2,MySQL error code与SQLSTATE并不是一一对应的,比如很多MySQL error code都映射到同一SQLSTATE code(HY000) 

当MySQL客户端碰到错误时,它会报告MySQL error code和相关的SQLSATE code:

mysql > CALL nosuch_sp();
ERROR 1305 (42000): PROCEDURE sqltune.nosuch_sp does not exist

具体的sqlsdate和mysql error code的对应可以在http://dev.mysql.com/doc/的MySQL reference manual的附录B找到完整的最新的error codes 

 

condition_name:命名条件

MySQL error code或者SQLSTATE code的可读性太差,所以引入了命名条件:

语法:

DECLARE condition_name CONDITION FOR condition_value

condition_value:
    SQLSTATE [VALUE] sqlstate_value
  | mysql_error_code

 使用: 

# original
DECLARE CONTINUE HANDLER FOR 1216 MySQL_statements;

# changed
DECLARE foreign_key_error CONDITION FOR 1216;
DECLARE CONTINUE HANDLER FOR foreign_key_error MySQL_statements;

  用condition_name为错误代码起了个别名。

 

示例1:Duplicate entry Handler

CREATE PROCEDURE sp_add_location
    (in_location    VARCHAR(30),
     in_address1    VARCHAR(30),
     in_address2    VARCHAR(30),
     zipcode        VARCHAR(10),
     OUT out_status VARCHAR(30))
BEGIN
    DECLARE CONTINUE HANDLER
        FOR 1062
        SET out_status='Duplicate Entry';

    SET out_status='OK';
    INSERT INTO locations
        (location,address1,address2,zipcode)
    VALUES
        (in_location,in_address1,in_address2,zipcode);
END;

 

示例2: Last Row Handler

 

CREATE PROCEDURE sp_not_found()
    READS SQL DATA
BEGIN
    DECLARE l_last_row INT DEFAULT 0;
    DECLARE l_dept_id INT:
    DECLARE c_dept CURSOR FOR
        SELECT department_id FROM departments;
    DECLARE CONTINUE HANDLER FOR NOT FOUND SET l_last_row=1;

    OPEN c_dept;
    dept_cursor: LOOP
        FETCH c_dept INTO l_dept_id;
        IF (l_last_row=1) THEN
            LEAVE dept_cursor;
        END IF;
    END LOOP dept_cursor;
    CLOSE c_dept;
END;

 

综合示例:

CREATE PROCEDURE sp_add_department
    (p_department_name     VARCHAR(30),
     p_manager_surname     VARCHAR(30),
     p_manager_firstname   VARCHAR(30),
     p_location            VARCHAR(30),
     OUT p_sqlcode         INT,
     OUT p_status_message  VARCHAR(100))
BEGIN

    /* START Declare Conditions */

    DECLARE duplicate_key CONDITION FOR 1062;
    DECLARE foreign_key_violated CONDITION FOR 1216;

    /* END Declare COnditions */

    /* START Declare variables and cursors */

    DECLARE l_manager_id INT;
    DECLARE csr_mgr_id CURSOR FOR
        SELECT employee_id FROM employees
        WHERE surname=UPPER(p_manager_surname)
        AND firstname=UPPER(p_manager_firstname);

    /* END Declare variables and cursors */

    /* START Declare Exception Handlers */

    DECLARE CONTINUE HANDLER FOR duplicate_key
    BEGIN
        SET p_sqlcode=1052;
        SET p_status_message='Duplicate key error';
    END;

    DECLARE CONTINUE HANDLER FOR foreign_key_violated
    BEGIN
        SET p_sqlcode=1216;
        SET p_status_message='Foreign key violated';
    END;

    DECLARE CONTINUE HANDLER FOR NOT FOUND
    BEGIN
        SET p_sqlcode=1329;
        SET p_status_message='No record found';
    END;

    /* END Declare Exception Handlers */

    /* START Execution */

    SET p_sqlcode=0;
    OPEN csr_mgr_id;
    FETCH csr_mgr_id INTO l_manager_id;

    IF p_sqlcode<>0 THEN     /* Failed to get manager id */
        SET p_status_message=CONCAT(p_status_message,' when fetching manager id');
    ELSE                     /* Got manager id, we can try and insert */
        INSERT INTO departments (department_name, manager_id, location)
        VALUES(UPPER(p_department_name), l_manager_id, UPPER(p_location));
        IF p_sqlcode<>0 THEN /* Failed to insert new department */
            SET p_status_message=CONCAT(p_status_message, ' when inserting new department');
        END IF;
    END IF;

    CLOSE csr_mgr_id;

    /* END Execution */

END 
分享到:
评论

相关推荐

    [MySQL] 存储过程错误异常处理例子

    总结来说,`DECLARE EXIT HANDLER FOR SQLEXCEPTION`是MySQL存储过程中处理错误和异常的关键工具。它使得我们能够优雅地处理可能出现的问题,确保即使在异常情况下,程序也能按照预定的方式进行操作,从而提升整个...

    mysql存储过程之错误处理实例详解

    MySQL存储过程中的错误处理是确保程序稳定性和可靠性的重要部分。在存储过程中,错误处理允许开发者预定义当特定错误出现时的响应方式,从而避免程序意外中断或者数据损坏。本篇文章将详细阐述如何在MySQL中进行存储...

    MySQL 存储过程

    ### MySQL存储过程详解 #### 一、MySQL存储过程概述 MySQL 存储过程是一种预编译的 SQL 代码块,可以包含复杂的逻辑控制结构。它能够接收输入参数、设置输出...希望本篇文章能帮助读者更好地理解和应用MySQL存储过程。

    MySQL实验报告5(存储过程与函数)(1)(1).pdf

    根据提供的文件内容,本篇实验报告主要围绕MySQL数据库中存储过程和函数的应用,涵盖了创建存储过程、函数、游标以及异常处理等高级特性。下面将详细解析报告中的每个知识点。 1. 创建存储过程 存储过程是一种在...

    Java调用数据库存储过程[mysql测试通过]

    以下是一个简单的Java调用MySQL存储过程的示例: ```java import java.sql.*; public class TestJavaProcedure { public static void main(String[] args) { String url = "jdbc:mysql://localhost:3306/mydb"; ...

    IBatis查删改查与调用存储过程 mysql数据库

    通过这篇文章,读者可以了解到如何在Java项目中使用IBatis进行数据库操作,并掌握调用MySQL存储过程的方法。作者很可能会提供详细的步骤、示例代码和实战案例,帮助读者更好地理解和应用这些技术。由于没有具体的...

    MySQL存储过程

    本篇文章将深入探讨MySQL存储过程的相关知识点。 1. **存储过程的定义与创建** - 存储过程是一组为了完成特定功能的SQL语句集合,经编译后存储在数据库中,当需要执行该功能时,只需要调用存储过程即可。 - 创建...

    jdbc调用mysql存储过程实现代码

    MySQL存储过程是一种预编译的SQL语句集合,可以在数据库端执行,提高应用程序的性能和效率。本篇文章将详细介绍如何使用JDBC(Java Database Connectivity)调用MySQL的存储过程。 1. **创建MySQL存储过程** 在...

    sql 存储过程的学习 和实例

    本篇文章将深入探讨SQL存储过程的学习与实例,以帮助你更好地理解和应用这一强大的数据库编程工具。 一、SQL存储过程的概念与优势 1. 存储过程是一种预编译的SQL语句集,它封装了复杂的业务逻辑,允许一次性提交多...

    JDBC 调用存储过程方法

    本篇文章将深入探讨如何使用JDBC调用存储过程。 首先,存储过程是预编译的SQL代码集合,它们封装在数据库中,可以接收输入参数,处理数据,然后返回结果或输出参数。存储过程在提高性能、增强安全性和简化复杂操作...

    存储过程的返回结果集有2中类型

    - **异常处理**:在存储过程中添加适当的错误处理机制,以便在出现问题时能提供有用的反馈。 - **测试与文档**:充分测试存储过程,确保其功能正确无误,并为每个过程编写清晰的文档,方便其他开发人员理解和使用。 ...

    C#中调用MySQL存储过程的方法

    本篇文章将详细讲解如何在C#中调用MySQL存储过程,同时涵盖一些数据库操作的基本技巧。 首先,我们需要创建一个`MySqlConnection`对象来建立与MySQL服务器的连接。这可以通过提供数据库名、服务器地址、用户名和...

    MySql数据库用java程序创建表以及存储过程.pdf

    在Java编程中,与MySQL数据库交互是常见的任务之一,其中包括创建数据库表和调用存储过程。本篇文章将详细讲解如何使用Java程序实现这些功能。 首先,要创建MySQL数据库中的表,我们可以利用Java的JDBC(Java ...

    mybatis+mysql 使用存储过程生成流水号的实现代码

    本篇文章将深入探讨如何在MyBatis框架中结合MySQL存储过程来生成流水号。 首先,存储过程`GetSerialNo`接受一个输入参数`tsCode`,类型为VARCHAR(50),并返回一个VARCHAR(200)类型的`result`。该过程的主要目的是为...

    调用存储过程

    3. 调用带有结果集的存储过程时,必须先处理结果集,否则可能会导致Connection关闭时抛出异常。 总结来说,Java调用存储过程是数据库操作中的一种常见技术,通过CallableStatement接口,我们可以方便地在Java程序中...

    james外网配置及使用MYSQL数据库存储邮件内容

    总之,配置James在外网运行并使用MySQL存储邮件内容,需要对服务器配置、数据库管理、邮件协议和网络安全有深入理解。这个过程涉及多个步骤,每个环节都关系到邮件服务的稳定性和安全性。正确完成这些步骤后,你可以...

    Python简单调用MySQL存储过程并获得返回值的方法

    本篇将详细介绍如何在Python中调用MySQL的存储过程,并获取其返回值。首先,我们需要安装`MySQLdb`库,这是Python连接MySQL数据库的一个流行库。 ### 安装MySQLdb库 在Python中,我们通常使用`pip`来安装`MySQLdb`...

    java调用存储过程

    本篇文章将详细讲解如何在Java中调用存储过程,特别是针对MySQL数据库。存储过程是预编译的SQL语句集合,它们提供了更好的性能,安全性和代码重用性。 首先,让我们了解什么是存储过程。存储过程是数据库中的一个...

    用C#编写的存储过程

    本篇文章将深入探讨如何用C#语言编写存储过程,并讨论其优点、适用场景以及如何在实际项目中应用。 存储过程是一种预编译的SQL代码集合,可以在数据库中存储并多次调用,提高了执行效率和代码复用性。传统上,存储...

    MySQL-进阶.pdf

    B+树索引是MySQL中最常用的索引类型之一,其特点是所有叶子节点都指向物理磁盘地址,这使得查找过程非常高效。 #### 三、性能调优 性能调优是MySQL进阶学习的重要部分,涉及到多个方面的调整和优化。 - **SQL语句...

Global site tag (gtag.js) - Google Analytics