MySQL 触发器
Posted woshihaiyong168
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了MySQL 触发器相关的知识,希望对你有一定的参考价值。
mysql 触发器:
大白话: 当某个表进行变动的时候 可以根据sql事件 对其他表做出修改 ; 远远没有事务强大!!!
建议少用 触发器 多使用存储过程!!
原因:
1.存储过程和触发器二者是有很大的联系的,我的一般理解就是触发器是一个隐藏的存储过程,因为它不需要参数,不需要显示调用,往往在你不知情的情况下已经做了很多操作。从这个角度来说,由于是隐藏的,无形中增加了系统的复杂性,非DBA人员理解起来数据库就会有困难,因为它不执行根本感觉不到它的存在。
2.再有,涉及到复杂的逻辑的时候,触发器的嵌套是避免不了的,如果再涉及几个存储过程,再加上事务等等,很容易出现死锁现象,再调试的时候也会经常性的从一个触发器转到另外一个,级联关系的不断追溯,很容易使人头大。其实,从性能上,触发器并没有提升多少性能,只是从代码上来说,可能在coding的时候很容易实现业务,所以我的观点是:摒弃触发器!触发器的功能基本都可以用存储过程来实现。
3.在编码中存储过程显示调用很容易阅读代码,触发器隐式调用容易被忽略。
存储过程也有他的致命伤
4.存储过程的致命伤在于移植性,存储过程不能跨库移植,比如事先是在mysql数据库的存储过程,考虑性能要移植到oracle上面那么所有的存储过程都需要被重写一遍。
直接上实例:
//跨库情况
库名:dingdphp
表1: d_user_serverid_info
CREATE TABLE `d_user_serverid_info` (
`uid` bigint(20) NOT NULL,
`serverId` int(11) NOT NULL,
PRIMARY KEY (`uid`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 ;
库名:log
表2:d_server_register
CREATE TABLE `d_server_register` (
`id` int(11) NOT NULL AUTO_INCREMENT COMMENT '自动编号',
`uid` bigint(20) NOT NULL DEFAULT '0' COMMENT '用户id',
`serverId` smallint(6) NOT NULL DEFAULT '0' COMMENT '服务器id',
`create_time` datetime DEFAULT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
需求 :
1、就是表1添加数据 表2 也需要添加一条
2、表1服务器id 修改 表2 也需要修改
建立触发器语句
use dingdphp; #是以表一为触发表
DELIMITER $$
DROP TRIGGER if exists d_user_serverid_info_insert;
CREATE TRIGGER d_user_serverid_info_insert
AFTER INSERT
ON dingdphp.d_user_serverid_info FOR EACH ROW
BEGIN
INSERT INTO log.d_server_register values(null,NEW.uid,NEW.serverId,now());
END
$$
DELIMITER ;
DELIMITER $$
DROP TRIGGER if exists d_user_serverid_info_update;
CREATE TRIGGER d_user_serverid_info_update
AFTER UPDATE
ON dingdphp.d_user_serverid_info FOR EACH ROW
BEGIN
update log.d_server_register set serverId=new.serverId,create_time=now() where uid=old.uid;
END
$$
DELIMITER ;
效果展示:
添加:
update
demo2:
没有和java方沟通好 他们插入d_user_serverid_info 表是运用的 replace into (存在即删除改变,不存在添加) replace into介绍
这个时候我依旧用的上面的触发器 导致数据库中出现了冗余数据(同一条数据有两天甚至三条)
这个 时候需要我们对d_server_register表重构
DROP TABLE if exists d_server_register;
CREATE TABLE `d_server_register` (
`uid` bigint(20) NOT NULL COMMENT 'uid',
`serverId` smallint(6) NOT NULL DEFAULT '0' COMMENT '服务器id',
`create_time` datetime NOT NULL COMMENT '创建时间',
PRIMARY KEY (`uid`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
因为不存在修改 所以我们删除掉update 触发器
DROP TRIGGER if exists d_user_serverid_info_update;
对插入触发器进行修改
DELIMITER $$
DROP TRIGGER if exists d_user_serverid_info_insert;
CREATE TRIGGER d_user_serverid_info_insert
AFTER INSERT
ON dingdphp.d_user_serverid_info FOR EACH ROW
BEGIN
REPLACE INTO ms_log.d_server_register values(NEW.uid,NEW.serverId,now());
END
$$
DELIMITER ;
这样就保持了 数据的唯一性!!!!!
详解:
MySQL包含对触发器的支持。触发器是一种与表操作有关的数据库对象,当触发器所在表上出现指定事件时,将调用该对象,即表的操作事件触发表上的触发器的执行。 创建触发器 在MySQL中,创建触发器语法如下: 代码如下:CREATE TRIGGER trigger_name trigger_time trigger_event ON tbl_name FOR EACH ROW trigger_stmt 其中: trigger_name:标识触发器名称,用户自行指定; trigger_time:标识触发时机,取值为 BEFORE 或 AFTER; trigger_event:标识触发事件,取值为 INSERT、UPDATE 或 DELETE; tbl_name:标识建立触发器的表名,即在哪张表上建立触发器; trigger_stmt:触发器程序体,可以是一句SQL语句,或者用 BEGIN 和 END 包含的多条语句。 由此可见,可以建立6种触发器,即:BEFORE INSERT、BEFORE UPDATE、BEFORE DELETE、AFTER INSERT、AFTER UPDATE、AFTER DELETE。 另外有一个限制是不能同时在一个表上建立2个相同类型的触发器,因此在一个表上最多建立6个触发器。 trigger_event 详解 MySQL 除了对 INSERT、UPDATE、DELETE 基本操作进行定义外,还定义了 LOAD DATA 和 REPLACE 语句,这两种语句也能引起上述6中类型的触发器的触发。 LOAD DATA 语句用于将一个文件装入到一个数据表中,相当与一系列的 INSERT 操作。 REPLACE 语句一般来说和 INSERT 语句很像,只是在表中有 primary key 或 unique 索引时,如果插入的数据和原来 primary key 或 unique 索引一致时,会先删除原来的数据,然后增加一条新数据,也就是说,一条 REPLACE 语句有时候等价于一条。 INSERT 语句,有时候等价于一条 DELETE 语句加上一条 INSERT 语句。 INSERT 型触发器:插入某一行时激活触发器,可能通过 INSERT、LOAD DATA、REPLACE 语句触发; UPDATE 型触发器:更改某一行时激活触发器,可能通过 UPDATE 语句触发; DELETE 型触发器:删除某一行时激活触发器,可能通过 DELETE、REPLACE 语句触发。 BEGIN … END 详解 在MySQL中,BEGIN … END 语句的语法为: BEGIN [statement_list] END 其中,statement_list 代表一个或多个语句的列表,列表内的每条语句都必须用分号(;)来结尾。 而在MySQL中,分号是语句结束的标识符,遇到分号表示该段语句已经结束,MySQL可以开始执行了。因此,解释器遇到statement_list 中的分号后就开始执行,然后会报出错误,因为没有找到和 BEGIN 匹配的 END。 这时就会用到 DELIMITER 命令(DELIMITER 是定界符,分隔符的意思),它是一条命令,不需要语句结束标识,语法为: DELIMITER new_delemiter new_delemiter 可以设为1个或多个长度的符号,默认的是分号(;),我们可以把它修改为其他符号,如$: DELIMITER $ 在这之后的语句,以分号结束,解释器不会有什么反应,只有遇到了$,才认为是语句结束。注意,使用完之后,我们还应该记得把它给修改回来。 一个完整的创建触发器示例 假设系统中有两个表: 班级表 class(班级号 classID, 班内学生数 stuCount) 学生表 student(学号 stuID, 所属班级号 classID) 要创建触发器来使班级表中的班内学生数随着学生的添加自动更新,代码如下: 代码如下: 下载地址 springmvc+mybatis+spring 整合SSM DELIMITER $ create trigger tri_stuInsert after insert on student for each row begin declare c int; set c = (select stuCount from class where classID=new.classID); update class set stuCount = c + 1 where classID = new.classID; end$ DELIMITER ; 变量详解 MySQL 中使用 DECLARE 来定义一局部变量,该变量只能在 BEGIN … END 复合语句中使用,并且应该定义在复合语句的开头, 即其它语句之前,语法如下: DECLARE var_name[,...] type [DEFAULT value] 其中: var_name 为变量名称,同 SQL 语句一样,变量名不区分大小写;type 为 MySQL 支持的任何数据类型;可以同时定义多个同类型的变量,用逗号隔开;变量初始值为 NULL,如果需要,可以使用 DEFAULT 子句提供默认值,值可以被指定为一个表达式。 对变量赋值采用 SET 语句,语法为: SET var_name = expr [,var_name = expr] ... NEW 与 OLD 详解 上述示例中使用了NEW关键字,和 MS SQL Server 中的 INSERTED 和 DELETED 类似,MySQL 中定义了 NEW 和 OLD,用来表示 触发器的所在表中,触发了触发器的那一行数据。 具体地: 在 INSERT 型触发器中,NEW 用来表示将要(BEFORE)或已经(AFTER)插入的新数据; 在 UPDATE 型触发器中,OLD 用来表示将要或已经被修改的原数据,NEW 用来表示将要或已经修改为的新数据; 在 DELETE 型触发器中,OLD 用来表示将要或已经被删除的原数据; 使用方法: NEW.columnName (columnName 为相应数据表某一列名) 另外,OLD 是只读的,而 NEW 则可以在触发器中使用 SET 赋值,这样不会再次触发触发器,造成循环调用(如每插入一个学生前,都在其学号前加“2013”)。 查看触发器 和查看数据库(show databases;)查看表格(show tables;)一样,查看触发器的语法如下: SHOW TRIGGERS [FROM schema_name]; 其中,schema_name 即 Schema 的名称,在 MySQL 中 Schema 和 Database 是一样的,也就是说,可以指定数据库名,这样就 不必先“USE database_name;”了。 删除触发器 和删除数据库、删除表格一样,删除触发器的语法如下: DROP TRIGGER [IF EXISTS] [schema_name.]trigger_name 触发器的执行顺序 我们建立的数据库一般都是 InnoDB 数据库,其上建立的表是事务性表,也就是事务安全的。这时,若SQL语句或触发器执行失败,MySQL 会回滚事务,有: ①如果 BEFORE 触发器执行失败,SQL 无法正确执行。 ②SQL 执行失败时,AFTER 型触发器不会触发。 ③AFTER 类型的触发器执行失败,SQL 会回滚
以上是关于MySQL 触发器的主要内容,如果未能解决你的问题,请参考以下文章
建立一个存储过程student_info,要求根据班级查询学生的学号、姓名、课程号和分数(表结构如表2,表3)