一起聊聊MySQL基础之触发器和事件

本篇文章给大家带来了关于mysql中触发器和事件的相关知识,其中包括触发器使用注意事项、查看和删除事件、事件使用注意事项等等,希望对大家有帮助。

一起聊聊MySQL基础之触发器和事件

触发器

我们使用MySQL的过程中可能会有下边这些需求:

在向t1表插入或更新数据之前对自动对数据进行校验,要求m1列的值必须在1~10之间,校验规则如下:

如果插入的记录的m1列的值小于1,则按1插入。如果m1列的值大于10,则按10插入。

在向t1表中插入记录之后自动把这条记录插入到t2表。

也就是我们在对表中的记录做增、删、改操作前和后都可能需要让MySQL服务器自动执行一些额外的语句,这个就是所谓的触发器的应用场景。

创建触发器

我们看一下定义触发器的语句:

CREATE TRIGGER 触发器名{BEFORE|AFTER}{INSERT|DELETE|UPDATE}ON 表名FOR EACH ROWBEGIN    触发器内容END

小贴士:由大括号`{}`包裹并且内部用竖线`|`分隔的语句表示必须在给定的选项中选取一个值,比如`{BEFORE|AFTER}`表示必须在`BEFORE`、`AFTER`这两个之间选取一个。

其中{BEFORE|AFTER}表示触发器内容执行的时机,它们的含义如下:

名称 描述

BEFORE表示在具体的语句执行之前就开始执行触发器的内容AFTER表示在具体的语句执行之后才开始执行触发器的内容

{INSERT|DELETE|UPDATE}表示具体的语句,MySQL中目前只支持对INSERTDELETEUPDATE这三种类型的语句设置触发器。

FOR EACH ROW BEGIN ... END表示对具体语句影响的每一条记录都执行我们自定义的触发器内容:

对于INSERT语句来说,FOR EACH ROW影响的记录就是我们准备插入的那些新记录。

对于DELETE语句和UPDATE语句来说,FOR EACH ROW影响的记录就是符合WHERE条件的那些记录(如果语句中没有WHERE条件,那就是代表全部的记录)。

小贴士: 如果触发器内容只包含一条语句,那也可以省略BEGN、END这两个词儿。

因为MySQL服务器会对某条语句影响的所有记录依次调用我们自定义的触发器内容,所以针对每一条受影响的记录,我们需要一种访问该记录中的内容的方式,MySQL提供了NEWOLD两个单词来分别代表新记录和旧记录,它们在不同语句中的含义不同:

对于INSERT语句设置的触发器来说,NEW代表准备插入的记录,OLD无效。对于DELETE语句设置的触发器来说,OLD代表删除前的记录,NEW无效。对于UPDATE语句设置的触发器来说,NEW代表修改后的记录,OLD代表修改前的记录。

现在我们可以正式定义一个触发器了:

mysql> delimiter $mysql> CREATE TRIGGER bi_t1    -> BEFORE INSERT ON t1    -> FOR EACH ROW    -> BEGIN    ->     IF NEW.m1          SET NEW.m1 = 1;    ->     ELSEIF NEW.m1 > 10 THEN    ->         SET NEW.m1 = 10;    ->     END IF;    -> END $Query OK, 0 rows affected (0.02 sec)mysql> delimiter ;mysql>

我们对t1表定义了一个名叫bi_t1触发器,它的意思就是在对t1表插入新记录之前,对准备插入的每一条记录都会执行BEGIN ... END之间的语句,NEW.列名表示当前待插入记录指定列的值。现在t1表中一共有4条记录:

mysql> SELECT * FROM t1;+------+------+| m1   | n1   |+------+------+|    1 | a    ||    2 | b    ||    3 | c    ||    4 | d    |+------+------+4 rows in set (0.00 sec)mysql>

我们现在执行一下插入语句并再次查看一下t1表的内容:

mysql> INSERT INTO t1(m1, n1) VALUES(5, 'e'), (100, 'z');Query OK, 2 rows affected (0.00 sec)Records: 2  Duplicates: 0  Warnings: 0mysql> SELECT * FROM t1;+------+------+| m1   | n1   |+------+------+|    1 | a    ||    2 | b    ||    3 | c    ||    4 | d    ||    5 | e    ||   10 | z    |+------+------+6 rows in set (0.00 sec)mysql>

这个INSERT语句影响的记录有两条,分别是(5, 'e')(100, 'z'),这两条记录将分别执行我们自定义的触发器内容。很显然(5, 'e')被成功的插入到了t1表中,而(100, 'z')插入到表中后却变成了(10, 'z'),这个就说明我们的bi_t1触发器生效了!

小贴士: 我们上边定义的触发器名`bi_t1`的`bi`是`before insert`的首字母缩写,`t1`是表名。虽然对于触发器的命名并没有什么特殊的要求,但是习惯上还是建议大家把它定义我上边例子中的形式,也就是`bi_表名`、`bd_表名`、`bu_表名`、`ai_表名`、`ad_表名`、`au_表名`的形式。

上边只是举了一个对INSERT语句设置BEFORE触发器的例子,对DELETEUPDATE操作设置BEFORE或者AFTER触发器的过程是类似的,就不赘述了。

查看和删除触发器

查看当前数据库中定义的所有触发器的语句:

SHOW TRIGGERS;

查看某个具体的触发器的定义:

SHOW CREATE TRIGGER 触发器名;

删除触发器:

DROP TRIGGER 触发器名;

这几个命令太简单了,就不举例子了啊~

触发器使用注意事项

触发器内容中不能有输出结果集的语句。

比方说:

mysql> delimiter $mysql> CREATE TRIGGER ai_t1    -> AFTER INSERT ON t1    -> FOR EACH ROW    -> BEGIN    ->     SELECT NEW.m1, NEW.n1;    -> END $ERROR 1415 (0A000): Not allowed to return a result set from a triggermysql>

显示的ERROR的意思就是不允许在触发器内容中返回结果集!

触发器内容中NEW代表记录的列的值可以被更改,OLD代表记录的列的值无法更改。

NEW代表新插入或着即将修改后的记录,修改它的列的值将影响INSERT和UPDATE语句执行后的结果,而OLD代表修改或删除之前的值,我们无法修改它。比方说如果我们非要这么写那就会报错的:

文心一言 文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

文心一言 1008 查看详情 文心一言

mysql> delimiter $mysql> CREATE TRIGGER bu_t1    -> BEFORE UPDATE ON t1    -> FOR EACH ROW    -> BEGIN    ->     SET OLD.m1 = 1;    -> END $ERROR 1362 (HY000): Updating of OLD row is not allowed in triggermysql>

可以看到提示的错误中显示在触发器中OLD代表的记录是不可被更改的。

在BEFORE触发器中,我们可以使用SET NEW.列名 = 某个值的形式来更改待插入记录或者待更新记录的某个列的值,但是这种操作不能在AFTER触发器中使用,因为在执行AFTER触发器的内容时记录已经被插入完成或者更新完成了。

比方说如果我们非要这么写那就会报错的:

mysql> delimiter $mysql>     CREATE TRIGGER ai_t1    ->     AFTER INSERT ON t1    ->     FOR EACH ROW    ->     BEGIN    ->         SET NEW.m1 = 1;    ->     END $ERROR 1362 (HY000): Updating of NEW row is not allowed in after triggermysql>

可以看到提示的错误中显示在AFTER触发器中是不允许更改NEW代表的记录的。

如果我们的BEFORE触发器内容执行过程中遇到了错误,那这个触发器对应的具体语句将无法执行;如果具体的操作语句执行过程中遇到了错误,那与它对应的AFTER触发器的内容将无法执行。

小贴士: 对于支持事务的表,不论是执行触发器内容还是具体操作语句过程中出现了错误,会把这个过程中所有的语句都回滚。当然,作为小白的我们并不知道啥是个事务,啥是个回滚,这些进阶内容都在《MySQL是怎样运行的:从根儿上理解MySQL》中呢~

事件

有时候我们想让MySQL服务器在某个时间点或者每隔一段时间自动地执行一些语句,这时候就需要去创建一个事件

创建事件

创建事件的语法如下:

CREATE EVENT 事件名ON SCHEDULE{    AT 某个确定的时间点|     EVERY 期望的时间间隔 [STARTS datetime][END datetime]}DOBEGIN    具体的语句END

事件支持两种类型的自动执行方式:

在某个确定的时间点执行。

比方说:

CREATE EVENT insert_t1_eventON SCHEDULEAT '2019-09-04 15:48:54'DOBEGIN    INSERT INTO t1(m1, n1) VALUES(6, 'f');END

我们在这个事件中指定了执行时间是'2019-09-04 15:48:54',除了直接填某个时间常量,我们也可以填写一些表达式:

CREATE EVENT insert_t1ON SCHEDULEAT DATE_ADD(NOW(), INTERVAL 2 DAY)DOBEGIN    INSERT INTO t1(m1, n1) VALUES(6, 'f');END

其中的DATE_ADD(NOW(), INTERVAL 2 DAY)表示该事件将在当前时间的两天后执行。

每隔一段时间执行一次。

比方说:

CREATE EVENT insert_t1ON SCHEDULEEVERY 1 HOURDOBEGIN    INSERT INTO t1(m1, n1) VALUES(6, 'f');END

其中的EVERY 1 HOUR表示该事件将每隔1个小时执行一次。默认情况下,采用这种每隔一段时间执行一次的方式将从创建事件的事件开始,无限制的执行下去。我们也可以指定该事件开始执行时间和截止时间:

CREATE EVENT insert_t1ON SCHEDULEEVERY 1 HOUR STARTS '2019-09-04 15:48:54' ENDS '2019-09-16 15:48:54'DOBEGIN    INSERT INTO t1(m1, n1) VALUES(6, 'f');END

如上所示,该事件将从’2019-09-04 15:48:54’开始直到’2019-09-16 15:48:54’为止,中间每隔1个小时执行一次。

小贴士: 表示事件间隔的单位除了HOUR,还可以用YEAR、QUARTER、MONTH、DAY、HOUR、 MINUTE、WEEK、SECOND、YEAR_MONTH、DAY_HOUR、DAY_MINUTE、DAY_SECOND、HOUR_MINUTE、HOUR_SECOND、MINUTE_SECOND这些单位,根据具体需求选用我们需要的时间间隔单位。

在创建好事件之后我们就不用管了,到了指定时间,MySQL服务器会帮我们自动执行的。

查看和删除事件

查看当前数据库中定义的所有事件的语句:

SHOW EVENTS;

查看某个具体的事件的定义:

SHOW CREATE EVENT 事件名;

删除事件:

DROP EVENT 事件名;

这几个命令太简单了,就不举例子了啊~

事件使用注意事项

默认情况下,MySQL服务器并不会帮助我们执行事件,除非我们使用下边的语句手动开启该功能:

mysql> SET GLOBAL event_scheduler = ON;Query OK, 0 rows affected (0.00 sec)mysql>

小贴士: event_scheduler其实是一个系统变量,它的值也可以在MySQL服务器启动的时候通过启动参数或者通过配置文件来设置event_scheduler的值。这些所谓的系统变量、启动参数、配置文件的各种东东并不是我们小白现在需要掌握的,大家忽略它们就好了~

推荐学习:mysql视频教程

以上就是一起聊聊MySQL基础之触发器和事件的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/284965.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月4日 18:01:51
下一篇 2025年11月4日 18:05:19

相关推荐

  • C++ 函数库在哪些场景下使用?

    c++++ 函数库预先定义了代码模块,可用于处理常见任务,广泛应用于:输入/输出操作字符串操作数学运算此外,还有特定领域的场景,如:图形处理网络编程数据库连接示例:使用 matplotlibcpp 函数库绘制折线图,实现了便捷的图形化展示。 C++ 函数库的应用场景 简介C++ 函数库提供了预定义的…

    2025年12月18日
    000
  • C++框架内置功能在跨平台开发中的作用

    在跨平台开发中,c++++ 框架的内置功能发挥着至关重要的作用,包括:跨平台兼容性:在多平台上稳定运行,简化移植。图形用户界面 (gui) 支持:提供跨平台的 gui 库,无需编写平台特定代码。数据库集成:支持多种数据库系统,实现数据访问和操作的跨平台性。网络通信:提供网络通信机制,用于分布式应用程…

    2025年12月18日
    000
  • C++框架中的数据处理和存储机制

    本文介绍了 c++++ 框架中处理和存储数据的关键技术:数据处理:stl 容器(vector、map、set)boost 库(额外容器和算法)eigen 库(优化矩阵和向量)数据存储:关系数据库管理系统(rdbms)(mysql、postgresql、sqlite)文档存储(mongodb、redi…

    2025年12月18日
    000
  • C++框架在行业中的实际案例

    c++++ 框架广泛应用于行业中,提供高效且可维护的代码解决方案。这些框架用于开发跨平台 gui(qt)、科学计算(armadillo、eigen)、线性代数运算(eigen)、计算机视觉(opencv)和高性能网络应用程序(ace)。例如,spotify 使用 boost(线程管理和网络操作)、a…

    2025年12月18日
    000
  • C++框架在后端开发方面的优缺点?

    c++++框架在后端开发中的优缺点:优点:高性能低内存使用跨平台原生数据库支持丰富的库生态缺点:学习曲线陡峭内存管理复杂缺乏动态关注编译时间长缺乏垃圾回收 C++ 框架在后端开发中的优缺点 优点: 高性能: C++ 是一种编译语言,可生成高效机器码,从而实现优异的性能。低内存使用: C++ 提供直接…

    2025年12月18日
    000
  • 在线资源和教程如何帮助降低 C++ 框架的学习难度?

    通过在线资源和教程,你可以降低 c++++ 框架的学习难度:利用官方文档和外部教程学习框架的基本概念和最佳实践。加入 stack overflow 和讨论论坛等社区,寻求具体问题的帮助和支持。通过构建 restful api、数据库应用程序和网络服务器等实战案例,获得实践经验。 通过在线资源和教程降…

    2025年12月18日
    000
  • 不同C++框架在特定领域的专业化和适用性对比

    不同的 c++++ 框架针对特定领域进行了专业化,以优化性能和易用性。对于 web 开发,boost.asio 以其轻量级和跨平台能力脱颖而出。用于 gui 开发的 qt 提供了丰富的功能和跨平台支持,而 qtsql 简化了数据库访问。poco 在网络编程中很突出,openmp 在并行计算中表现出色…

    2025年12月18日
    000
  • C++ 框架如何优化大型项目中资源管理和性能

    c++++ 框架通过资源管理和性能优化,提升了大型项目的效率:资源管理优化:通过资源池和智能指针管理资源,优化分配和释放操作。性能优化:利用并发编程和内存管理库,实现跨平台并发编程和提升内存管理性能。 C++ 框架:优化大型项目中资源管理和性能 大型 C++ 项目经常面临资源管理和性能问题。为了解决…

    2025年12月18日
    000
  • C++ 框架性能基准:与其他语言和平台的比较

    基准测试结果表明,c++++ 框架在 http 请求处理和数据库查询方面表现优异,boost.asio 适用于高并发连接,cpp-httplib 和 libuv 具有较低延迟。开发人员应根据不同场景选择最合适的框架。 C++ 框架性能基准:与其他语言和平台的比较 引言 性能是任何软件应用程序的关键方…

    2025年12月18日
    000
  • C++ 框架在网络安全领域的应用:提升网络抵御能力与威胁防御

    c++++ 框架在网络安全中广泛应用,提升网络抵御和威胁防御能力:高性能:c++ 速度和效率使其适合构建实时网络安全应用。内存管理:c++ 提供对内存的细粒度控制,确保安全和资源高效。跨平台支持:c++ 可在多种平台编译,便于构建跨平台安全解决方案。应用包括:入侵检测系统、防火墙、恶意软件分析和取证…

    2025年12月18日
    000
  • C++ 框架的学习指南:快速上手

    对于使用 c++++ 框架,本指南提供了分步指导,包括:了解基础知识:熟悉框架概念和 oop 原则。探索框架功能:容器、算法、线程管理、网络操作和数据库集成。实战案例:创建一个文件管理程序,演示框架的使用。扩展知识:探索其他框架、参与社区和练习项目。 C++ 框架的学习指南:快速上手 随着 C++ …

    2025年12月18日
    000
  • C++ 框架设计中实现代码重用的技术

    在 c++++ 框架设计中,实现代码重用的技术包括:模板方法模式:定义算法大纲,由子类定义具体步骤。策略模式:分离算法实现和使用对象,提高灵活性。工厂方法模式:创建对象的方法由子类实现,允许创建不同类型对象。抽象工厂模式:创建相关对象家族的方法,无需指定具体类,促进松耦合。单例模式:确保类只有一个实…

    2025年12月18日
    000
  • c++是什么语言

    C++ 是一种通用、面向对象、跨平台的编程语言,由比雅尼·斯特劳斯特鲁普开发于 1985 年。它具有面向对象、跨平台、高效、可扩展和低级控制等特点。C++ 被广泛应用于开发操作系统、数据库管理系统、游戏引擎、网络应用程序和大型企业软件。其优点包括效率高、可移植性好、面向对象和强大;缺点则包括复杂性、…

    2025年12月18日
    000
  • C++ 开源库指南:探索丰富的库资源,拓展编程能力

    c++++开源库为开发者提供了丰富的工具,包括:boost库:提供算法和数据结构,可用于字符串操作和图表处理。eigen:线性代数库。opencv:计算机视觉和机器学习库。qt:跨平台gui框架。mysql connector:与mysql数据库交互的库。wxwidgets:图形用户界面库。选择最合…

    2025年12月18日
    000
  • c++怎么连接数据库

    在C++中,可以使用ODBC或第三方库连接到数据库。ODBC步骤:初始化环境,连接到DBMS。其他库:MySQL Connector/C++、PostgreSQL libpq++、SQLite3,根据数据库类型和需求选择库。 C++连接数据库 在C++中连接数据库,可以使用标准的C++数据库连接库(…

    2025年12月18日
    000
  • 在 C++ 生态系统中,如何选择合适的库和框架?

    在 c++++ 生态系统中进行库和框架选择时,应考虑功能、性能、稳定性、社区支持和许可证等因素。实战案例包括 qt(图形库)、rapidjson(json 库)、boost.asio(网络库)和 mysql-connector-cpp(数据库连接库)。使用库管理器和仔细研究文档有助于做出明智的选择。…

    2025年12月18日
    000
  • 学习C++ Web开发所需的技能和资源有哪些?

    c++++ web 开发需要掌握 c++ 编程基础、网络协议和数据库知识。必备资源包括 cppcms、pistache 等 web 框架,cppdb、pqxx 等数据库连接器,以及 cmake、g++、wireshark 等辅助工具。通过学习实战案例,如创建简单的 http 服务器,可以开启 c++…

    2025年12月18日
    000
  • c++属于前端还是后端

    C++ 既可用于前端(用户界面创建)也可用于后端(服务器端逻辑处理),因为它强大且高效,支持多范例编程。决定因素包括应用类型、性能要求和开发团队技能。 C++:前端还是后端? C++ 既可以用于前端开发,也可以用于后端开发。因此,它不属于前端或后端的单一类别。 前端开发 涉及用户界面(UI)的创建,…

    2025年12月18日
    000
  • Dapper如何封装通用仓储 Dapper Repository模式实现方法

    Dapper通用仓储应借鉴EF思想而非照搬,核心是泛型约束+手写SQL灵活性:定义IRepository接口(GetById/Find/Insert/Update/Delete),实现类通过特性识别主键与列映射,动态生成安全SQL,支持事务参数,分页由具体方法处理,查询逻辑下沉至具体仓储,连接由DI…

    2025年12月17日
    000
  • Dapper中[Key]和[Table]特性有什么用 Dapper.Contrib特性详解

    [Table] 和 [Key] 是 Dapper.Contrib 中用于实体映射的核心特性:[Table] 标注类以指定对应数据库表名,避免默认复数推导错误;[Key] 标注自增主键属性,插入时忽略该字段并回填生成值;非自增主键须用[ExplicitKey]。 [Table] 和 [Key] 是 D…

    2025年12月17日
    000

发表回复

登录后才能评论
关注微信