Maison > base de données > tutoriel mysql > Parlons des bases des déclencheurs et événements MySQL

Parlons des bases des déclencheurs et événements MySQL

WBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWB
Libérer: 2022-01-20 18:07:51
avant
2367 Les gens l'ont consulté

Cet article vous apporte des connaissances pertinentes sur les déclencheurs et les événements dans MySQL, y compris les précautions d'utilisation des déclencheurs, l'affichage et la suppression d'événements, les précautions d'utilisation des événements, etc. J'espère qu'il sera utile à tout le monde.

Parlons des bases des déclencheurs et événements MySQL

Déclencheurs

Nous pouvons avoir les exigences suivantes lors de l'utilisation de 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
Copier après la connexion

小贴士:

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

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

Insérer ou insérer dans la table t1 Avant de mettre à jour le data, les données sont automatiquement vérifiées. La valeur de la colonne m1 doit être comprise entre 1~10 Les règles de vérification sont les suivantes : Après avoir inséré un enregistrement dans la table t1, insérez automatiquement cet enregistrement dans la table t2. Autrement dit, nous devrons peut-être demander au serveur MySQL d'exécuter automatiquement certaines instructions supplémentaires avant et après l'ajout, la suppression ou la modification d'enregistrements dans la table. est ce qu'on appelle le scénario d'application de trigger.
名称 描述
BEFORE 表示在具体的语句执行之前就开始执行触发器的内容
AFTER
  • Si. la valeur de la colonne m1 de l'enregistrement inséré est inférieure à 1, insérez-la comme 1.
  • Si la valeur de la colonne m1 est supérieure à 10, insérez comme 10.
  • Créer un déclencheur

    🎜Jetons un coup d'œil à l'instruction qui définit un trigger : 🎜
    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>
    Copier après la connexion
    Copier après la connexion
    🎜Conseils :
    Une instruction entourée d'accolades `{}` et séparée par une barre verticale `|` signifie qu'une valeur doit être sélectionnée parmi les options données, comme `{BEFORE|AFTER}` signifie qu'elle doit être comprise entre ` AVANT`, `APRÈS`Choisissez-en un entre ces deux. 🎜
    🎜où {BEFORE|AFTER> représente le moment de l'exécution du contenu du déclencheur, et leurs significations sont les suivantes : 🎜 Nom🎜🎜AVANT🎜🎜Indique que le contenu du déclencheur sera exécuté avant l'exécution spécifique l'instruction est exécutée🎜 🎜🎜AFTER🎜🎜 signifie que le contenu du déclencheur ne sera pas exécuté tant que l'instruction spécifique n'est pas exécutée🎜🎜🎜🎜

    {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 < 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>
    Copier après la connexion
    Copier après la connexion

    我们对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>
    Copier après la connexion

    我们现在执行一下插入语句并再次查看一下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>
    Copier après la connexion

    这个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;
    Copier après la connexion

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

    SHOW CREATE TRIGGER 触发器名;
    Copier après la connexion

    删除触发器:

    DROP TRIGGER 触发器名;
    Copier après la connexion

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

    触发器使用注意事项

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

      比方说:

      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>
      Copier après la connexion

      显示的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>
      Copier après la connexion

      可以看到提示的错误中显示在触发器中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>
      Copier après la connexion

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

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

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

    事件

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

    创建事件

    创建事件的语法如下:

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

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

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

      比方说:

      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
      Copier après la connexion

      我们在这个事件中指定了执行时间是'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
      Copier après la connexion

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

    2. 每隔一段时间执行一次。

      比方说:

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

      其中的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
      Copier après la connexion

      如上所示,该事件将从'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;
    Copier après la connexion

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

    SHOW CREATE EVENT 事件名;
    Copier après la connexion

    删除事件:

    DROP EVENT 事件名;
    Copier après la connexion

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

    事件使用注意事项

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

    mysql> SET GLOBAL event_scheduler = ON;
    Query OK, 0 rows affected (0.00 sec)
    
    mysql>
    Copier après la connexion

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

    推荐学习:mysql视频教程

    Description

    Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

    Étiquettes associées:
    source:csdn.net
    Déclaration de ce site Web
    Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn
    Tutoriels populaires
    Plus>
    Derniers téléchargements
    Plus>
    effets Web
    Code source du site Web
    Matériel du site Web
    Modèle frontal