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是对所有没有被SQLWARNING或NOT 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
分享到:
相关推荐
总结来说,`DECLARE EXIT HANDLER FOR SQLEXCEPTION`是MySQL存储过程中处理错误和异常的关键工具。它使得我们能够优雅地处理可能出现的问题,确保即使在异常情况下,程序也能按照预定的方式进行操作,从而提升整个...
MySQL存储过程中的错误处理是确保程序稳定性和可靠性的重要部分。在存储过程中,错误处理允许开发者预定义当特定错误出现时的响应方式,从而避免程序意外中断或者数据损坏。本篇文章将详细阐述如何在MySQL中进行存储...
### MySQL存储过程详解 #### 一、MySQL存储过程概述 MySQL 存储过程是一种预编译的 SQL 代码块,可以包含复杂的逻辑控制结构。它能够接收输入参数、设置输出...希望本篇文章能帮助读者更好地理解和应用MySQL存储过程。
根据提供的文件内容,本篇实验报告主要围绕MySQL数据库中存储过程和函数的应用,涵盖了创建存储过程、函数、游标以及异常处理等高级特性。下面将详细解析报告中的每个知识点。 1. 创建存储过程 存储过程是一种在...
以下是一个简单的Java调用MySQL存储过程的示例: ```java import java.sql.*; public class TestJavaProcedure { public static void main(String[] args) { String url = "jdbc:mysql://localhost:3306/mydb"; ...
通过这篇文章,读者可以了解到如何在Java项目中使用IBatis进行数据库操作,并掌握调用MySQL存储过程的方法。作者很可能会提供详细的步骤、示例代码和实战案例,帮助读者更好地理解和应用这些技术。由于没有具体的...
本篇文章将深入探讨MySQL存储过程的相关知识点。 1. **存储过程的定义与创建** - 存储过程是一组为了完成特定功能的SQL语句集合,经编译后存储在数据库中,当需要执行该功能时,只需要调用存储过程即可。 - 创建...
MySQL存储过程是一种预编译的SQL语句集合,可以在数据库端执行,提高应用程序的性能和效率。本篇文章将详细介绍如何使用JDBC(Java Database Connectivity)调用MySQL的存储过程。 1. **创建MySQL存储过程** 在...
本篇文章将深入探讨SQL存储过程的学习与实例,以帮助你更好地理解和应用这一强大的数据库编程工具。 一、SQL存储过程的概念与优势 1. 存储过程是一种预编译的SQL语句集,它封装了复杂的业务逻辑,允许一次性提交多...
本篇文章将深入探讨如何使用JDBC调用存储过程。 首先,存储过程是预编译的SQL代码集合,它们封装在数据库中,可以接收输入参数,处理数据,然后返回结果或输出参数。存储过程在提高性能、增强安全性和简化复杂操作...
- **异常处理**:在存储过程中添加适当的错误处理机制,以便在出现问题时能提供有用的反馈。 - **测试与文档**:充分测试存储过程,确保其功能正确无误,并为每个过程编写清晰的文档,方便其他开发人员理解和使用。 ...
本篇文章将详细讲解如何在C#中调用MySQL存储过程,同时涵盖一些数据库操作的基本技巧。 首先,我们需要创建一个`MySqlConnection`对象来建立与MySQL服务器的连接。这可以通过提供数据库名、服务器地址、用户名和...
在Java编程中,与MySQL数据库交互是常见的任务之一,其中包括创建数据库表和调用存储过程。本篇文章将详细讲解如何使用Java程序实现这些功能。 首先,要创建MySQL数据库中的表,我们可以利用Java的JDBC(Java ...
本篇文章将深入探讨如何在MyBatis框架中结合MySQL存储过程来生成流水号。 首先,存储过程`GetSerialNo`接受一个输入参数`tsCode`,类型为VARCHAR(50),并返回一个VARCHAR(200)类型的`result`。该过程的主要目的是为...
3. 调用带有结果集的存储过程时,必须先处理结果集,否则可能会导致Connection关闭时抛出异常。 总结来说,Java调用存储过程是数据库操作中的一种常见技术,通过CallableStatement接口,我们可以方便地在Java程序中...
总之,配置James在外网运行并使用MySQL存储邮件内容,需要对服务器配置、数据库管理、邮件协议和网络安全有深入理解。这个过程涉及多个步骤,每个环节都关系到邮件服务的稳定性和安全性。正确完成这些步骤后,你可以...
本篇将详细介绍如何在Python中调用MySQL的存储过程,并获取其返回值。首先,我们需要安装`MySQLdb`库,这是Python连接MySQL数据库的一个流行库。 ### 安装MySQLdb库 在Python中,我们通常使用`pip`来安装`MySQLdb`...
本篇文章将详细讲解如何在Java中调用存储过程,特别是针对MySQL数据库。存储过程是预编译的SQL语句集合,它们提供了更好的性能,安全性和代码重用性。 首先,让我们了解什么是存储过程。存储过程是数据库中的一个...
本篇文章将深入探讨如何用C#语言编写存储过程,并讨论其优点、适用场景以及如何在实际项目中应用。 存储过程是一种预编译的SQL代码集合,可以在数据库中存储并多次调用,提高了执行效率和代码复用性。传统上,存储...
6. **使用存储过程**: 将复杂的业务逻辑封装到存储过程中,减少网络往返,提高性能。 7. **数据分区**: 对于非常大的表,可以考虑使用分区策略,如范围分区、列表分区等,以提高数据插入和查询速度。 8. **避免...