mysql触发器支持哪些操作_mysql触发事件说明

3次阅读

mysql触发器仅响应insert、update、delete行级dml操作,不支持truncate、drop、alter等ddl及视图/临时表操作;每表最多6个触发器(before/after各事件一个);old/new为上下文引用,非变量,作用域限于触发器内。

mysql触发器支持哪些操作_mysql触发事件说明

mysql触发器只响应 INSERT、UPDATE、DELETE 三类操作

MySQL 触发器**不支持**对 TRUNCATE tableDROP TABLEALTER TABLE 或视图/临时表的操作;它只在明确的行级 DML 语句执行时激活。这是硬性限制,不是配置问题。

  • INSERT:包括 INSERT INTO ... VALUESINSERT INTO ... selectREPLACE(会触发 DELETE + INSERT)、LOAD DATA
  • UPDATE:仅当某行实际被修改(字段值真正变化)时才触发;若 SET x=x 这种无变更更新,默认不触发(除非启用了 sql_mode 中的 STRICT_TRANS_TABLES严格模式
  • DELETE:包括普通 DELETE FROMREPLACE 中的删除阶段;但分区表的 ALTER TABLE ... DROP PARTITION 不会激活 DELETE 触发器

BEFORE 和 AFTER 的本质区别不是“时机早晚”,而是“能否改写数据”

关键判断标准是:你是否需要干预即将写入或刚写入的数据。这直接决定该用 BEFORE 还是 AFTER

  • BEFORE INSERT/UPDATE:可用 NEW.column 修改待插入/更新的值(如自动填充时间、校验并重设非法值);NEW 可写,OLDINSERT 中不可用
  • BEFORE DELETE:可用 OLD.column 读取将被删的行,但无法阻止删除(要用 signal 抛异常中断)
  • AFTER 类触发器:NEWOLD 都只读;适合日志记录、跨表同步、调用存储过程等“事后动作”;不能修改当前语句影响的行

常见错误:想在 AFTER UPDATE 里用 SET NEW.amount = ... —— 会报错 Can't update table in stored function/trigger

每个表最多 6 个触发器,且不能复用事件+时间组合

MySQL 对单表触发器数量有硬上限:每种「事件+时间」组合只能有一个触发器。也就是说,一个表最多允许:

  • 1 个 BEFORE INSERT
  • 1 个 AFTER INSERT
  • 1 个 BEFORE UPDATE
  • 1 个 AFTER UPDATE
  • 1 个 BEFORE DELETE
  • 1 个 AFTER DELETE

试图再建第二个 BEFORE INSERT 会直接报错:Error 1359 (HY000): Trigger already exists。解决办法只有合并逻辑到一个触发器中,或用 BEGIN...END 块内多语句处理。

别名 OLDNEW 不是变量,是只读/只写上下文引用

它们不是用户定义变量,不能 SET @var = OLD.id 后再用(虽然语法不报错,但行为不可靠),更不能 DECLARE。它们的作用域仅限于当前触发器体,且访问规则严格:

  • INSERT:只有 NEW 可用,代表新行;OLD 不存在
  • DELETE:只有 OLD 可用,代表被删行;NEW 不存在
  • UPDATEOLD 是更新前的值,NEW 是更新后的值;两者都存在,但 OLD 只读,NEWBEFORE 中可写

典型误用:在 AFTER UPDATE 中尝试 UPDATE other_table SET x = NEW.y —— 表面可行,但若该语句又触发另一个触发器,可能引发递归或死锁,且 MySQL 默认禁用触发器嵌套(max_sp_recursion_depth=0)。

触发器逻辑越靠近数据层,就越难调试和测试;尤其是涉及多表写入或调用函数时,事务边界和错误传播路径容易模糊。上线前务必验证 SIGNAL 异常是否真能回滚主语句,以及 AFTER 触发器失败是否会导致整个事务失败(默认是的)。

text=ZqhQzanResources