mysql的触发器是什么级的

mysql触发器是行级的。按照SQL标准,触发器可以分为两种:1、行级触发器,对于修改的每一行数据都会激活一次,如果一个语句插入了100行数据,将会调用触发器100次;2、语句级触发器,针对每个语句激活一次,一个插入100行数据的语句只会调用一次触发器。而MySQL中只支持行级触发器,不支持预语句级触发器。

mysql的触发器是什么级的

本教程操作环境:windows7系统、mysql8版本、Dell G3电脑。

触发器概述

MySQL 触发器(trigger)是一种存储程序,它和一个指定的表相关联,当该表中的数据发生变化(增加、更新、删除)时自动执行。 这些修改数据行的操作被称为触发器事件,例如 INSERT 或者 LOAD DATA 等插入数据的语句可以激活插入触发器。

按照 SQL 标准,触发器可以分为行级触发器(row-level trigger)和语句级触发器( statement-level trigger)。

行级触发器对于修改的每一行数据都会激活一次,如果一个语句插入了 100 行数据,将会调用触发器 100 次;

语句级触发器针对每个语句激活一次,一个插入 100 行数据的语句只会调用一次触发器。

MySQL 只支持行级触发器,不支持预语句级触发器。

1.png

不同事件可以激活不同类型的触发器。INSERT 事件触发器用于插入数据的操作,包括 INSERT、LOAD DATA、REPLACE 语句等;UPDATE 事件触发器用于更新操作,例如 UPDATE 语句;DELETE 事件触发器用于删除操作,例如 DELETE 和 REPLACE 语句等,DROP TABLE 和 TRUNCATE TABLE 语句不会激活删除触发器。

另外,MySQL 触发器可以在触发事件之前或者之后执行,分别称为 BEFORE 触发器和 AFTER 触发器。这两种触发时机可以和不同的触发事件进行组合,例如 BEFORE INSERT 触发器或者 AFTER UPDATE 触发器。

MySQL 触发器的优点包括:

记录并审核用户对表中数据的修改操作,实现审计功能;

实现比检查约束更复杂的完整性约束,例如禁止非业务时间的数据操作;

实现某种业务逻辑,例如增加或删除员工时自动更新部门中的人数;

同步实时地复制表中的数据。

虽然触发器功能强大,但是它也存在一些缺点:

触发器会增加数据库结构的复杂度,而且触发器对应用程序不可见,难以调试;

触发器需要占用更多的数据库服务器资源,尽量使用数据库提供的非空、唯一、检查约束等;

触发器不能接收参数,只能基于当前的触发对象进行操作。

针对特殊场景使用触发器可以带来一定的便利性;但不要过渡依赖触发器,避免造成数据库的性能下降和维护困难。接下来我们介绍触发器的管理操作。

MySQL 支持的三种触发器

在实际使用中,MySQL 所支持的触发器有三种:INSERT 触发器、UPDATE 触发器和 DELETE 触发器。

1) INSERT 触发器

在 INSERT 语句执行之前或之后响应的触发器。

使用 INSERT 触发器需要注意以下几点:

在 INSERT 触发器代码内,可引用一个名为 NEW(不区分大小写)的虚拟表来访问被插入的行。

在 BEFORE INSERT 触发器中,NEW 中的值也可以被更新,即允许更改被插入的值(只要具有对应的操作权限)。

对于 AUTO_INCREMENT 列,NEW 在 INSERT 执行之前包含的值是 0,在 INSERT 执行之后将包含新的自动生成值。

2) UPDATE 触发器

在 UPDATE 语句执行之前或之后响应的触发器。

使用 UPDATE 触发器需要注意以下几点:

在 UPDATE 触发器代码内,可引用一个名为 NEW(不区分大小写)的虚拟表来访问更新的值。

在 UPDATE 触发器代码内,可引用一个名为 OLD(不区分大小写)的虚拟表来访问 UPDATE 语句执行前的值。

在 BEFORE UPDATE 触发器中,NEW 中的值可能也被更新,即允许更改将要用于 UPDATE 语句中的值(只要具有对应的操作权限)。

OLD 中的值全部是只读的,不能被更新。

PPT.CN,PPTCN,PPT.CN是什么,PPT.CN官网,PPT.CN如何使用 PPT.CN,PPTCN,PPT.CN是什么,PPT.CN官网,PPT.CN如何使用

一键操作,智能生成专业级PPT

PPT.CN,PPTCN,PPT.CN是什么,PPT.CN官网,PPT.CN如何使用 37 查看详情 PPT.CN,PPTCN,PPT.CN是什么,PPT.CN官网,PPT.CN如何使用

注意:当触发器设计对触发表自身的更新操作时,只能使用 BEFORE 类型的触发器,AFTER 类型的触发器将不被允许。

3) DELETE 触发器

在 DELETE 语句执行之前或之后响应的触发器。

使用 DELETE 触发器需要注意以下几点:

在 DELETE 触发器代码内,可以引用一个名为 OLD(不区分大小写)的虚拟表来访问被删除的行。

OLD 中的值全部是只读的,不能被更新。

总体来说,触发器使用的过程中,MySQL 会按照以下方式来处理错误。

对于事务性表,如果触发程序失败,以及由此导致的整个语句失败,那么该语句所执行的所有更改将回滚;对于非事务性表,则不能执行此类回滚,即使语句失败,失败之前所做的任何更改依然有效。

若 BEFORE 触发程序失败,则 MySQL 将不执行相应行上的操作。

若在 BEFORE 或 AFTER 触发程序的执行过程中出现错误,则将导致调用触发程序的整个语句失败。

仅当 BEFORE 触发程序和行操作均已被成功执行,MySQL 才会执行 AFTER 触发程序。

创建触发器

MySQL 使用CREATE TRIGGRT语句创建触发器,基本语法如下:

CREATE TRIGGER trigger_name    { BEFORE | AFTER } { INSERT | UPDATE | DELETE }    ON table_name FOR EACH ROW    trigger_body;

其中,trigger_name 是触发器的名称;BEFORE 和 AFTER 用于指定触发器的触发时机;INSERT、UPDATE 和 DELETE 用于定义触发事件的类型;table_name 是触发器关联的表名,不能是临时表或者视图;FOR EACH ROW 表明这是一个行级触发器;trigger_body 是触发器执行的具体语句。

举例来说,由于员工的薪水属于重要信息,所以需要记录薪水的修改历史。首先,我们创建一个审计表:

CREATE TABLE emp_salary_audit (    audit_id    INTEGER NOT NULL AUTO_INCREMENT    emp_id      INTEGER NOT NULL,    old_salary  NUMERIC(8,2) NULL,    new_salary  NUMERIC(8,2) NULL,    change_date TIMESTAMP NOT NULL,    change_by   VARCHAR(50) NOT NULL,    CONSTRAINT pk_emp_salary_audit PRIMARY KEY (audit_id));

其中,audit_id 是自增主键;emp_id 是员工编号;old_salary 和 new_salary 分别用于存储修改前和修改后的月薪;change_date 记录了修改时间;change_by 记录了执行修改操作的用户。

然后创建一个触发器 tri_audit_salary,用于记录员工月薪的修改记录:

DELIMITER $$CREATE TRIGGER tri_audit_salary  AFTER UPDATE ON employee  FOR EACH ROWBEGIN  -- 当月薪改变时,记录审计数据  IF (NEW.salary  OLD.salary) THEN   INSERT INTO salary_audit (emp_id, old_salary, new_salary, change_date, change_by)   VALUES(OLD.emp_id, OLD.salary, NEW.salary, CURRENT_TIMESTAMP, USER());  END IF;END$$DELIMITER ;

其中,DELIMITER 用于修改 SQL 语句的结束符,我们在前文介绍存储过程时已经有所了解;AFTER 表示在修改数据之后执行触发器;UPDATE 表示只针对更新操作记录数据变化;触发器主体中的 NEW 和 OLD 是 MySQL 触发器中的特殊变量,包含了修改后和修改前的记录,对于 INSERT 触发器而言没有 OLD 变量,对于 DELETE 触发器而言没有 NEW 变量;CURRENT_TIMESTAMP 和 USER() 都是 MySQL 系统函数,返回当前时间和登录的用户。

创建触发器之后,我们执行一些数据修改的操作,验证该触发器的效果:

UPDATE employeeSET email = 'sunqian@shuguo.net'WHERE emp_name = '孙乾';UPDATE employeeSET salary = salary * 1.1WHERE emp_name = '孙乾';SELECT *FROM salary_audit;audit_id|emp_id|old_salary|new_salary|change_date        |change_by|--------|------|----------|----------|-------------------|---------|       1|    25|      4700|      5170|2019-10-18 10:16:36|TONY     |

第一个 UPDATE 语句只修改了“孙乾”的电子邮箱,所以不会触发 tri_audit_salary;第二个 UPDATE 语句修改了他的月薪,触发了 tri_audit_salary。因此审计表 salary_audit 中包含一条数据,记录了月薪变化前后的情况。

如果想要同时审计新增员工和删除员工的操作,可以再创建一个 INSERT 触发器和 DELETE 触发器。

除此之外,MySQL 支持针对相同的触发时机和相同的事件定义多个触发器,同时指定它们的执行顺序:

CREATE TRIGGER trigger_name    { BEFORE | AFTER } { INSERT | UPDATE | DELETE }    ON table_name FOR EACH ROW    { FOLLOWS | PRECEDES } other_trigger    trigger_body;

其中,FOLLOWS 表示该触发器在触发器 other_trigger 之后执行;PRECEDES 表示该触发器在 other_trigger 之前执行;如果没有指定任何选项,默认情况下按照触发器的创建顺序执行。

查看触发器

使用SHOW TRIGGERS语句可以查看数据库中的触发器列表:

SHOW TRIGGERS    [{FROM | IN} db_name]    [LIKE 'pattern' | WHERE expr]

其中,db_name 用于查看指定数据库中的触发器,默认为当前数据库;LIKE 用于匹配存储过程的名称,WHERE 可以指定更多的过滤条件。例如,以下语句返回了当前数据库中的触发器:

mysql> show triggersG*************************** 1. row ***************************             Trigger: tri_audit_salary               Event: UPDATE               Table: employee           Statement: BEGIN  -- 当月薪改变时,记录审计数据  IF (NEW.salary  OLD.salary) THEN   INSERT INTO salary_audit (emp_id, old_salary, new_salary, change_date, change_by)   VALUES(OLD.emp_id, OLD.salary, NEW.salary, CURRENT_TIMESTAMP, USER());  END IF;END              Timing: AFTER             Created: 2020-10-06 21:50:02.47            sql_mode: STRICT_TRANS_TABLES,NO_ENGINE_SUBSTITUTION             Definer: root@localhostcharacter_set_client: utf8mb4collation_connection: utf8mb4_0900_ai_ci  Database Collation: utf8mb4_0900_ai_ci1 row in set (0.00 sec)

另外,MySQL 系统表 INFORMATION_SCHEMA.TRIGGERS 中包含了更详细的触发器信息。

如果想要获取创建某个触发器的 DDL 语句,可以SHOW CREATE TRIGGER语句。例如:

mysql> SHOW CREATE TRIGGER tri_audit_salaryG*************************** 1. row ***************************               Trigger: tri_audit_salary              sql_mode: STRICT_TRANS_TABLES,NO_ENGINE_SUBSTITUTIONSQL Original Statement: CREATE DEFINER=`root`@`localhost` TRIGGER `tri_audit_salary` AFTER UPDATE ON `employee` FOR EACH ROW BEGIN  -- 当月薪改变时,记录审计数据  IF (NEW.salary  OLD.salary) THEN   INSERT INTO salary_audit (emp_id, old_salary, new_salary, change_date, change_by)   VALUES(OLD.emp_id, OLD.salary, NEW.salary, CURRENT_TIMESTAMP, USER());  END IF;END  character_set_client: utf8mb4  collation_connection: utf8mb4_0900_ai_ci    Database Collation: utf8mb4_0900_ai_ci               Created: 2020-10-06 21:50:02.471 row in set (0.00 sec)

删除触发器

MySQL 没有提供修改触发器的语句,只能通过DROP TRIGGER语句删除并再次创建触发器。例如,以下语句可以用于删除触发器 tri_audit_salary:

DROP TRIGGER IF EXISTS tri_audit_salary;

IF EXISTS 可以避免触发器 tri_audit_salary 不存在时产生错误。

【相关推荐:mysql视频教程】

以上就是mysql的触发器是什么级的的详细内容,更多请关注创想鸟其它相关文章!

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

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月4日 15:27:01
下一篇 2025年11月4日 15:32:04

相关推荐

  • Golang 如何实现一个在线留言墙_Golang Web 表单与数据持久化实战

    答案是使用Golang可快速实现留言墙。通过net/http处理路由与表单,定义Message结构体存储数据,内存切片暂存消息,ParseForm解析POST请求,Go模板渲染页面,支持用户提交与展示留言,基础功能完整,适合学习与原型开发。 想用 Golang 实现一个简单的在线留言墙?其实并不复杂…

    2025年12月16日
    000
  • 如何用Golang实现微服务配置中心_Golang 配置中心开发与实践

    使用Golang基于Etcd可构建轻量级配置中心,支持配置存储、动态更新、多环境管理、版本控制和访问控制;通过HTTP接口实现配置获取与监听,客户端利用goroutine实时感知变更,结合缓存、批量查询与健康检查优化生产可用性。 在微服务架构中,配置中心是核心组件之一。它解决了服务配置分散、变更困难…

    2025年12月16日
    000
  • 如何在Golang中实现云原生配置中心服务

    使用Golang结合Etcd实现云原生配置中心,支持动态更新、多环境隔离与版本控制。通过clientv3操作Etcd存储配置,提供RESTful API供客户端获取和监听配置,利用Watch机制实现实时推送,配合本地缓存与回调函数实现配置热加载,构建轻量高效的服务解耦方案。 在云原生架构中,配置中心…

    2025年12月16日
    000
  • Golang如何实现简单的内容管理后台

    答案:用Golang标准库net/http搭建Web服务,定义Post结构体实现内容模型,通过内存切片存储数据并提供RESTful API接口,使用%ignore_a_1%/template渲染前端页面,配合JavaScript调用API完成增删改查,初期可忽略安全机制,后期需添加认证与防护措施。 …

    2025年12月16日
    000
  • Golang结构体中的指针字段有什么作用_Golang结构体引用类型应用实例

    指针字段用于共享数据、节省内存和实现可变性。通过指向同一地址,多个结构体实例可共享底层数据,避免复制开销并实现同步更新;在大型对象或配置共享场景中提升性能;支持nil状态,适用于延迟初始化,如数据库连接池懒加载;合理使用可优化性能与状态管理。 在Go语言中,结构体的指针字段主要用于共享数据、节省内存…

    2025年12月16日
    000
  • Go语言中内嵌方法访问“父”字段的机制探讨

    在go语言中,内嵌结构体的方法无法直接访问其外部(“父”)结构体的字段,因为方法的接收者明确是内嵌类型,不持有外部结构体的引用。本文将深入探讨这一机制,解释其背后的原理,并提供两种解决思路:通过显式传递“父”引用作为替代方案,以及更符合go惯用法的api设计,即采用外部函数或服务对象来处理数据持久化…

    2025年12月16日
    000
  • Go语言中内嵌结构体方法访问宿主字段的机制与实践

    在go语言中,内嵌结构体的方法无法直接访问其宿主(父级)结构体的字段或方法,因为方法的接收者类型是固定的,不具备宿主上下文。本文将深入探讨这一机制,并通过代码示例验证其局限性,同时提供一种通过接口引用宿主的间接解决方案,并最终建议采用更符合go语言习惯的api设计模式,即分离数据和操作,以实现更清晰…

    2025年12月16日
    000
  • Go语言中嵌入类型方法访问“父”字段的机制与最佳实践

    在go语言中,嵌入类型的方法无法直接访问其宿主(“父”)结构体的非嵌入字段。这是因为嵌入机制是类型提升而非继承,方法的接收器始终是其声明时的类型。本文将深入探讨这一限制的原因,并提供两种解决方案:一种是手动传递“父”引用(不推荐),另一种是重新思考api设计,采用更符合go惯例的显式依赖方式,如db…

    2025年12月16日
    000
  • 深入理解Go语言嵌入:方法与宿主结构体字段的访问机制

    Go语言中,嵌入类型的方法接收者是嵌入类型本身,而非其宿主(embedding)结构体。这意味着嵌入方法无法直接访问宿主结构体的非嵌入字段。若需实现类似功能,可考虑在嵌入类型中引入一个接口字段来引用宿主,但这会增加复杂性。更推荐的设计模式是采用 `db.Save(user)` 形式的函数式API,以…

    2025年12月16日
    000
  • Go语言中将MySQL数据高效转换为JSON的实用指南

    本文旨在提供一个在go语言中将mysql数据库表数据高效转换为json格式的教程。我们将探讨在处理数据库扫描结果时,如何避免所有数据类型都被识别为字节数组(`[]byte`)的问题,并通过动态类型检查和适当的数据转换,确保数值、布尔值等原始类型在json中得到正确表示,从而生成符合预期的json输出…

    2025年12月16日
    000
  • Go database/sql:深度解析预处理语句与直接查询的机制与实践

    go 语言 `database/sql` 包为数据库操作提供了通用接口。本文深入探讨了预处理语句与直接查询在 go 中的实现机制,阐明了驱动在参数化处理中的核心作用。我们将解释为何 `query/queryrow` 也能安全接受参数,并分析预处理语句在防范 sql 注入、提升重复查询性能方面的独特优…

    2025年12月16日
    000
  • Go语言数据库操作:深入理解预处理语句与直接查询

    Go语言的`database/sql`包提供了一个高度抽象的数据库接口,其设计旨在兼容各种SQL数据库系统。无论是使用预处理语句还是直接查询并传递参数,`database/sql`包都通过底层的数据库驱动程序确保了参数的安全性处理,有效防止SQL注入。预处理语句尤其适用于高频重复执行的查询,能通过数…

    2025年12月16日
    000
  • 深入理解 Go database/sql:预处理语句的价值与参数处理机制

    go语言的 `database/sql` 包提供了一个通用的数据库接口,抽象了底层sql数据库的差异。尽管 `query` 和 `queryrow` 等方法允许直接传递参数,看似与预处理语句功能等价,但其内部实现仍依赖数据库驱动进行安全的参数转义和处理。预处理语句通过两步式操作(准备与执行),赋予驱…

    2025年12月16日
    000
  • Golang MySQL连接:正确指定数据库的实践指南

    本教程旨在解决%ignore_a_1%使用`go-sql-driver/mysql`连接mysql时,因尝试通过`use`语句选择数据库而导致的“no database selected”错误。核心内容是指导开发者应在dsn(data source name)中直接指定目标数据库,而非在连接建立后执…

    2025年12月16日
    000
  • Go语言中将IP地址等二进制数据存储到MySQL的BINARY类型字段

    在Go语言中,将IP地址等二进制数据存储到MySQL的`BINARY(N)`类型字段时,直接传递`[]byte`或`[N]byte`会遇到类型转换错误。本文将详细介绍如何通过将`[]byte`数据类型显式转换为`string`类型,利用`go-sql-driver/mysql`驱动的特性,有效解决这…

    2025年12月16日
    000
  • 如何在Go语言中高效地将二进制数据存储到MySQL的BINARY列

    本文详细阐述了在Go语言中使用`database/sql`驱动向MySQL的`BINARY(X)`列存储二进制数据(如IP地址)的有效方法。针对直接传递`[]byte`或`[4]byte`导致的类型转换错误,文章提出了将`net.IP`切片显式转换为`string`的解决方案,并提供了示例代码,确保…

    2025年12月16日
    000
  • Go语言与MySQL:高效存储二进制IP地址到BINARY字段

    在使用go语言的`go-sql-driver/mysql`驱动程序将二进制ip地址(如`net.ip`或`[]byte`)存储到mysql的`binary(4)`字段时,直接传递这些类型常会导致错误。本教程将详细介绍如何通过将`[]byte`类型的ip地址显式转换为`string`类型来解决这一问题…

    2025年12月16日
    000
  • Go语言与MySQL:正确存储二进制IP地址数据

    在go语言中,将二进制ip地址(如`net.ip.to4()`返回的`[]byte`)存储到mysql的`binary(4)`类型字段时,直接传递`[4]byte`数组或`net.ip`切片会导致类型转换错误。本文将详细探讨常见的存储误区,并提供一种简洁有效的解决方案:通过将`[]byte`切片显式…

    2025年12月16日
    000
  • Go语言中利用构建约束实现App Engine与标准SQL环境的条件编译

    本文将指导如何在go语言项目中,通过使用构建约束(`// +build` directives)优雅地解决google app engine (gae) 特定包(如`appengine/cloudsql`)与标准sql库在不同环境下的兼容性问题。我们将探讨如何利用`appengine`和`!appe…

    2025年12月16日
    000
  • Golang如何使用工厂模式创建对象

    Go语言通过接口和结构体实现工厂模式,封装对象创建过程。定义Database接口及MySQL、PostgreSQL实现,工厂函数NewDatabase根据类型返回对应实例,支持扩展与配置,提升代码可维护性。 在Go语言中,工厂模式通过函数或方法封装对象的创建过程,避免重复代码,提升可维护性。虽然Go…

    2025年12月16日
    000

发表回复

登录后才能评论
关注微信