侯体宗的博客
  • 首页
  • Hyperf版
  • beego仿版
  • 人生(杂谈)
  • 技术
  • 关于我
  • 更多分类
    • 文件下载
    • 文字修仙
    • 中国象棋ai
    • 群聊
    • 九宫格抽奖
    • 拼图
    • 消消乐
    • 相册

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

数据库  /  管理员 发布于 6年前   303

本文实例讲述了mysql存储过程之错误处理。分享给大家供大家参考,具体如下:

当存储过程中发生错误时,重要的是适当处理它,例如:继续或退出当前代码块的执行,并发出有意义的错误消息。其中mysql提供了一种简单的方法来定义处理从一般条件(如警告或异常)到特定条件(例如特定错误代码)的处理程序。完事我们来使用DECLARE HANDLER语句来尝试声明一个处理程序,先来看语法:

DECLARE action HANDLER FOR condition_value statement;

上述sql中,如果条件的值与condition_value匹配,则MySQL将执行statement,并根据该操作继续或退出当前的代码块。其中,操作(action)接受以下值之一:

  • CONTINUE:继续执行封闭代码块(BEGIN ... END)。
  • EXIT:处理程序声明封闭代码块的执行终止。

condition_value指定一个特定条件或一类激活处理程序的条件。condition_value接受以下值之一:

  • 一个MySQL错误代码。
  • 标准SQLSTATE值或者它可以是SQLWARNING,NOTFOUND或SQLEXCEPTION条件,这是SQLSTATE值类的简写。NOTFOUND条件用于游标或SELECT INTO variable_list语句。
  • 与MySQL错误代码或SQLSTATE值相关联的命名条件。

最重要的是,上述sql可以是一个简单的语句或由BEGIN和END关键字包围的复合语句。介绍完事之后,咱们来看几个声明处理程序的例子,首先是当程序发生错误时,将has_error变量的值设置为1并继续执行的例子:

DECLARE CONTINUE HANDLER FOR SQLEXCEPTION SET has_error = 1;

再来看当发生错误时,回滚上一个操作,发出错误消息,并退出当前代码块。 如果在存储过程的BEGIN END块中声明它,则会立即终止存储过程:

DECLARE EXIT HANDLER FOR SQLEXCEPTIONBEGINROLLBACK;SELECT 'An error has occurred, operation rollbacked and the stored procedure was terminated';END;

以下处理程序的意思是,如果没有更多的行要提取,在光标或select into语句的情况下,将no_row_found变量的值设置为1并继续执行:

DECLARE CONTINUE HANDLER FOR NOT FOUND SET no_row_found = 1;

以下处理程序如果发生重复的键错误,则会发出MySQL错误1062。 它发出错误消息并继续执行:

DECLARE CONTINUE HANDLER FOR 1062SELECT 'Error, duplicate key occurred';

上面这些实例可能有点抽象,咱们废话不多说,先来创建一个名为article_tags的新表,来具体操作下:

USE testdb;CREATE TABLE article_tags(  article_id INT,  tag_id   INT,  PRIMARY KEY(article_id,tag_id));

其中呢,article_tags表存储文章和标签之间的关系。每篇文章可能有很多标签,反之亦然。 为了简单起见,我们不会在article_tags表中创建文章(article)表和标签(tags)表以及外键。

完事呢,我们来创建一个存储过程,将文章的id和标签的id插入到article_tags表中:

USE testdb;DELIMITER $$CREATE PROCEDURE insert_article_tags(IN article_id INT, IN tag_id INT)BEGIN DECLARE CONTINUE HANDLER FOR 1062 SELECT CONCAT('duplicate keys (',article_id,',',tag_id,') found') AS msg; -- insert a new record into article_tags INSERT INTO article_tags(article_id,tag_id) VALUES(article_id,tag_id); -- return tag count for the article SELECT COUNT(*) FROM article_tags;END$$DELIMITER ;

然后呢,我们通过调用insert_article_tags存储过程,为文章ID为1添加标签ID:1,2和3,如下所示:

CALL insert_article_tags(1,1);CALL insert_article_tags(1,2);CALL insert_article_tags(1,3);

我们再尝试插入一个重复的键来检查处理程序是否真的被调用:

CALL insert_article_tags(1,3);

执行上面查询语句,得到以下结果:

mysql> CALL insert_article_tags(1,3);+----------------------------+| msg|+----------------------------+| duplicate keys (1,3) found |+----------------------------+1 row in set+----------+| COUNT(*) |+----------+|    3 |+----------+1 row in setQuery OK, 0 rows affected

执行后会收到一条错误消息。 但是,由于我们将处理程序声明为CONTINUE处理程序,所以存储过程继续执行。因此,最后获得了文章的标签计数值为:3。来看个图:

但是如果将处理程序声明中的CONTINUE更改为EXIT,那么将只会收到一条错误消息。如下查询语句:

DELIMITER $$CREATE PROCEDURE insert_article_tags_exit(IN article_id INT, IN tag_id INT)BEGIN DECLARE EXIT HANDLER FOR SQLEXCEPTION  SELECT 'SQLException invoked'; DECLARE EXIT HANDLER FOR 1062     SELECT 'MySQL error code 1062 invoked'; DECLARE EXIT HANDLER FOR SQLSTATE '23000' SELECT 'SQLSTATE 23000 invoked'; -- insert a new record into article_tags INSERT INTO article_tags(article_id,tag_id)  VALUES(article_id,tag_id); -- return tag count for the article SELECT COUNT(*) FROM article_tags;END $$DELIMITER ;

执行上面查询语句,得到以下结果:

mysql> CALL insert_article_tags_exit(1,3);+-------------------------------+| MySQL error code 1062 invoked |+-------------------------------+| MySQL error code 1062 invoked |+-------------------------------+1 row in setQuery OK, 0 rows affected

来看个图感受下:

如果我们使用多个处理程序来处理错误,MySQL将调用最特定的处理程序来处理错误。这就涉及到优先级的问题了,我们来具体看下。

我们知道错误总是映射到一个MySQL错误代码,因为在MySQL中它是最具体的。 SQLSTATE可以映射到许多MySQL错误代码,因此它不太具体。 SQLEXCPETION或SQLWARNING是SQLSTATES类型值的缩写,因此它是最通用的。假设在insert_article_tags_3存储过程中声明三个处理程序,如下所示:

DELIMITER $$CREATE PROCEDURE insert_article_tags_3(IN article_id INT, IN tag_id INT)BEGIN DECLARE EXIT HANDLER FOR 1062 SELECT 'Duplicate keys error encountered'; DECLARE EXIT HANDLER FOR SQLEXCEPTION SELECT 'SQLException encountered'; DECLARE EXIT HANDLER FOR SQLSTATE '23000' SELECT 'SQLSTATE 23000'; -- insert a new record into article_tags INSERT INTO article_tags(article_id,tag_id) VALUES(article_id,tag_id); -- return tag count for the article SELECT COUNT(*) FROM article_tags;END $$DELIMITER ;

然后我们尝试通过调用存储过程将重复的键插入到article_tags表中:

CALL insert_article_tags_3(1,3);

如下,可以看到MySQL错误代码处理程序被调用:

mysql> CALL insert_article_tags_3(1,3);+----------------------------------+| Duplicate keys error encountered |+----------------------------------+| Duplicate keys error encountered |+----------------------------------+1 row in setQuery OK, 0 rows affected

完事之后,咱们再来看下使用命名错误条件。从错误处理程序声明开始,如下:

DECLARE EXIT HANDLER FOR 1051 SELECT 'Please create table abc first';SELECT * FROM abc;

1051号是什么意思? 想象一下,你有一个大的存储过程代码使用了好多类似这样的数字; 这将成为维护代码的噩梦。幸运的是,MySQL为我们提供了声明一个命名错误条件的DECLARE CONDITION语句,它与条件相关联。DECLARE CONDITION语句的语法如下:

DECLARE condition_name CONDITION FOR condition_value;

condition_value可以是MySQL错误代码,例如:1015或SQLSTATE值。 condition_value由condition_name表示。声明后,可以参考condition_name,而不是参考condition_value。所以可以重写上面的代码如下:

DECLARE table_not_found CONDITION for 1051;DECLARE EXIT HANDLER FOR table_not_found SELECT 'Please create table abc first';SELECT * FROM abc;

这段代码比以前的代码显然更可读,不过我们要注意,条件声明必须出现在处理程序或游标声明之前。

好啦,这次就到这里了。

更多关于MySQL相关内容感兴趣的读者可查看本站专题:《MySQL存储过程技巧大全》、《MySQL常用函数大汇总》、《MySQL日志操作技巧大全》、《MySQL事务操作技巧汇总》及《MySQL数据库锁相关技巧汇总》

希望本文所述对大家MySQL数据库计有所帮助。


  • 上一条:
    MySQL密码正确却无法本地登录-1045
    下一条:
    阿里云ESC 安装 MYSQL8.0的教程
  • 昵称:

    邮箱:

    0条评论 (评论内容有缓存机制,请悉知!)
    最新最热
    • 分类目录
    • 人生(杂谈)
    • 技术
    • linux
    • Java
    • php
    • 框架(架构)
    • 前端
    • ThinkPHP
    • 数据库
    • 微信(小程序)
    • Laravel
    • Redis
    • Docker
    • Go
    • swoole
    • Windows
    • Python
    • 苹果(mac/ios)
    • 相关文章
    • 分库分表的目的、优缺点及具体实现方式介绍(0个评论)
    • DevDB - 在 VS 代码中直接访问数据库(0个评论)
    • 在ubuntu系统中实现mysql数据存储目录迁移流程步骤(0个评论)
    • 在mysql中使用存储过程批量新增测试数据流程步骤(0个评论)
    • php+mysql数据库批量根据条件快速更新、连表更新sql实现(0个评论)
    • 近期文章
    • 在go语言中使用api.geonames.org接口实现根据国际邮政编码获取地址信息功能(1个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf分页文件功能(0个评论)
    • gmail发邮件报错:534 5.7.9 Application-specific password required...解决方案(0个评论)
    • 欧盟关于强迫劳动的规定的官方举报渠道及官方举报网站(0个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf文件功能(0个评论)
    • Laravel从Accel获得5700万美元A轮融资(0个评论)
    • 在go + gin中gorm实现指定搜索/区间搜索分页列表功能接口实例(0个评论)
    • 在go语言中实现IP/CIDR的ip和netmask互转及IP段形式互转及ip是否存在IP/CIDR(0个评论)
    • PHP 8.4 Alpha 1现已发布!(0个评论)
    • Laravel 11.15版本发布 - Eloquent Builder中添加的泛型(0个评论)
    • 近期评论
    • 122 在

      学历:一种延缓就业设计,生活需求下的权衡之选中评论 工作几年后,报名考研了,到现在还没认真学习备考,迷茫中。作为一名北漂互联网打工人..
    • 123 在

      Clash for Windows作者删库跑路了,github已404中评论 按理说只要你在国内,所有的流量进出都在监控范围内,不管你怎么隐藏也没用,想搞你分..
    • 原梓番博客 在

      在Laravel框架中使用模型Model分表最简单的方法中评论 好久好久都没看友情链接申请了,今天刚看,已经添加。..
    • 博主 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 @1111老铁这个不行了,可以看看近期评论的其他文章..
    • 1111 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 网站不能打开,博主百忙中能否发个APP下载链接,佛跳墙或极光..
    • 2017-06
    • 2017-08
    • 2017-09
    • 2017-10
    • 2017-11
    • 2018-01
    • 2018-05
    • 2018-10
    • 2018-11
    • 2020-02
    • 2020-03
    • 2020-04
    • 2020-05
    • 2020-06
    • 2020-07
    • 2020-08
    • 2020-09
    • 2021-02
    • 2021-04
    • 2021-07
    • 2021-08
    • 2021-11
    • 2021-12
    • 2022-02
    • 2022-03
    • 2022-05
    • 2022-06
    • 2022-07
    • 2022-08
    • 2022-09
    • 2022-10
    • 2022-11
    • 2022-12
    • 2023-01
    • 2023-03
    • 2023-04
    • 2023-05
    • 2023-07
    • 2023-08
    • 2023-10
    • 2023-11
    • 2023-12
    • 2024-01
    • 2024-03
    Top

    Copyright·© 2019 侯体宗版权所有· 粤ICP备20027696号 PHP交流群

    侯体宗的博客