MySQL触发器的使用
触发器可以在执行语句前或执行后触发其他 SQL 代码运行。触发器可以读取触发语句改变了哪些数据,但是没有返回值。因此可以使用触发器加强业务逻辑的约束而不需要在应用程序写对应的代码。
从上述描述可以看到,触发器可以简化应用程序的逻辑并且可以提升性能,这是因为使用触发器减少了应用程序和服务端的交互次数。同时,触发器有助于完成自动更新归一化和统计数据。例如,我们可以使用触发器自动统计交易订单总金额,订单数及平均客单价。 然而,MySQL 的触发器的应用场合也十分有限,如果你使用过其他数据库产品的触发器,不要以为 MySQL 也能实现相同的功能,例如:
- 每个数据表的单一事件只能有一个触发器,也就是说对于 AFTER INSERT 这样的事件来说,不能同时有超过1个的触发器。
- MySQL 只支持行级别的触发器,也就是只能按 FOR EACH ROW 这种方式使用触发而不是整个 SQL 语句,这对于大量数据的操作而言会比较低效。MySQL 的触发器只能按下面的形式编写:
CREATE TRIGGER 触发器名 BEFORE|AFTER 触发事件ON 表名 FOR EACH ROWBEGIN 执行语句列表;END
DELIMITER $$CREATE TRIGGER user_create_log AFTER INSERT ON t_users FOR EACH ROWBEGINDECLARE log_info VARCHAR(40)character set utf8;DECLARE description VARCHAR(20) character set utf8;#后面发现中文字符编码出现乱码,这里设置字符集SET description = " is created";SET log_info = CONCAT(NEW.user_name, description); #函数CONCAT可以将字符串连接INSERT INTO logs(log) values(log_info);END $$DELIMITER ;
InnoDB 的触发器相关的操作,包括源语句都在同一个事务中,因此是满足原子性的。然而,如果使用InnoDB 的触发器去与另一张表校验数据一致性的时候,这个时候如果不小心的话可能导致不正确的结果。例如,假设需要使用触发器模拟外键,可以使用 BEFORE INSERT触发器验证另一张表是否存在对应的记录,但是如果在触发器读取另一张表数据的时候不使用 SELECT FOR UPDATE的话,则由于并发性性问题可能导致错误的结果。 虽然触发器有些缺陷,但是这并不意味着不能用。相反,触发器本身也是有用的,尤其是对于约束,系统维护任务和保持统计数据保持最新。
也可以使用触发器记录数据行的变化。这样即便是离线手动操作数据库的记录(如修复错误数据)也能够被记录下来。但是,需要注意的是对于往其他自增主键表插入数据时要小心,这对于复制性的语句表现会有问题,因为自增值对于两个相同的副本值并不同。
结语:
触发器在有限的场合能够发挥其优势,比如统计数据、数据表变更日志等。但是也会有一些缺陷,比如大数据量的更新由于逐行触发,会降低效率。还有就是,MyISAM 引擎无法保障原子性。因此,要根据应用场景是否要是有触发器。
更多相关文章
- MySQL系列多表连接查询92及99语法示例详解教程
- Linux下MYSQL 5.7 找回root密码的问题(亲测可用)
- MySQL 什么时候使用INNER JOIN 或 LEFT JOIN
- android从服务器下载文件(php+apache+win7+MySql)
- 【有图】android通过jdbc连接mysql(附文件)
- Android初始化语言 (init.*.rc、init.conf文件格式)
- Android初始化语言 (init.*.rc、init.conf文件格式)
- 如何写Android(安卓)init.rc
- Android(安卓)监听短信2种方式:Broadcast和ContentObserver