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

来源:这里教程网 时间:2026-02-28 15:10:29 作者:

本篇文章给大家带来了关于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 ROW
BEGIN
    触发器内容
END

其中

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

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

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

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

对于

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

对于

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

因为

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

对于
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 < 1 THEN
    ->         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: 0
mysql> 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
触发器生效了!

上边只是举了一个对

INSERT
语句设置
BEFORE
触发器的例子,对
DELETE
UPDATE
操作设置
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 trigger
mysql>

显示的

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

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

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

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 trigger
mysql>

可以看到提示的错误中显示在触发器中

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 trigger
mysql>

可以看到提示的错误中显示在AFTER触发器中是不允许更改

NEW
代表的记录的。

如果我们的

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

事件

有时候我们想让

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

创建事件

创建事件的语法如下:

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

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

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

    比方说:

    CREATE EVENT insert_t1_event
    ON SCHEDULE
    AT '2019-09-04 15:48:54'
    DO
    BEGIN
        INSERT INTO t1(m1, n1) VALUES(6, 'f');
    END

    我们在这个

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

    CREATE EVENT insert_t1
    ON SCHEDULE
    AT DATE_ADD(NOW(), INTERVAL 2 DAY)
    DO
    BEGIN
        INSERT INTO t1(m1, n1) VALUES(6, 'f');
    END

    其中的

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

    每隔一段时间执行一次。

    比方说:

    CREATE EVENT insert_t1
    ON SCHEDULE
    EVERY 1 HOUR
    DO
    BEGIN
        INSERT INTO t1(m1, n1) VALUES(6, 'f');
    END

    其中的

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

    CREATE EVENT insert_t1
    ON SCHEDULE
    EVERY 1 HOUR STARTS '2019-09-04 15:48:54' ENDS '2019-09-16 15:48:54'
    DO
    BEGIN
        INSERT INTO t1(m1, n1) VALUES(6, 'f');
    END

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

在创建好

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

查看和删除事件

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

SHOW EVENTS;

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

SHOW CREATE EVENT 事件名;

删除事件:

DROP EVENT 事件名;

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

事件使用注意事项

默认情况下,

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

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

推荐学习:mysql视频教程

相关推荐