1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
-----/*触发器*/
--满足触发条件时就会。自动执行。触发器中的语句,可以保证某些操作之间的一致性
--可以层叠更改,可以引用其他表中的列
--事前触发器可以获取事前之前和新的字段值,验证一些条件和进行一些准备操作,在表保存之前触发*/
--事后触发器是进行收尾工作,保证事务的完整性,在经表修改之后才能生效*/
--行级触发器是对DML语句影响的每个行执行一次,如UPDATE语句影响多行,就会对每行都激活一次触发器。
-----创建
--创建只有一个执行语句的触发器
 基本形式如下:
 create trigger 触发器名 before|after 触发事件 --触发事件可以为insert,update,delete
 on 表名 for each row 执行语句 
 例:
 create trigger dept_trig1 before insert
 on department for each row
 insert into trigger_time values(now());
 --补坑,创建trigger_time表
 drop table if exists tigger_time;--拼错了,少了一个'r'
 create table trigger_time(
  exec_time time
   )
 desc trigger_time
 --检查trigger效果
 select from department;
 insert into department values(1004,'销售部','负责产品销售','1号楼销售大厅')--报错,duplicate entry
 desc department --d_name是唯一性约束,不能取名‘销售部’
 insert into department values(1004,'销售1部','负责产品销售','1号楼销售大厅')
 select table_schema from information_schema.tables where table_name='trigger_time' --查看trigger_time表属于什么库
 select from trigger_time --查看trigger_time表是否被触发器更新,Ok 
  
 --创建有多个执行语句的触发器
 基本形式如下
 create trigger 触发器名 before|after 触发事件 --触发事件可以为insert,update,delete
   on 表名 for each row --在指定表上逐行触发
   begin
   执行语句1; 
   执行语句2; 
   .........;
   END
 /*MySQL默认以';'最为整段执行语句结束标志,这里用delimiter &&进行转意成&&,&&表示整段语句结束,在IDE里面应该可以不用转意*/
 例:
 delimiter && 
 create trigger dept_trigzz after delete
 on department for each row
 begin
 insert into trigger_time values('21:01:01');
 insert into trigger_time values('22:01:01');
 end
 &&
 delimiter;
 show triggers
  
 --检查trigger效果
 delete from department where d_id=1004;
 select from trigger_time--成功insert,ok
  
/*一个表在相同触发时间的相同触发事件,只能创建一个触发器*/
/*如department表中,触发事件Insert,触发事件为after的触发器只能有一个,但可以定义触发事件为Before的触发器*/
---查看触发器
---1.show triggers
show triggers --show triggers无法查询指定的触发器,只用于触发器较少的情况
---2.从triggers表中查看触发器信息
select from information_schema.triggers where tigger_name='触发器名'
---触发器的使用
--分别创建before insert和after insert两个触发器,比较执行顺序
--创建before insert触发器
create trigger before_insert before insert
on department for each row
insert into trigger_test values(null,'before_insert');
--创建after insert触发器
create trigger after_insert after insert
on department for each row
insert into trigger_test values(null,'after_insert');
--补坑,创建trigger_test表
create table trigger_test(id int auto_increment primary key,
info varchar(10)
);
--验证触发器
insert into department values(1004,'销售1部','负责产品销售','1号楼销售大厅');
delete from department where id = '1004'
desc department
delete from department where d_id = '1004'
insert into department values(1004,'销售1部','负责产品销售','1号楼销售大厅');--info字段报错,长度不够
--修改表中列的字段长度,
alter table trigger_test change column info info varchar(15); --change可以同时改表字段名与数据类型,但只修改字段长度时需重复一遍字段名
alter table trigger_test modify column info varchar(16);--modify只可以更改字段数据类型
--删除产生错误的插入列
delete from department where d_id = '1004'
--重新插入
insert into department values(1004,'销售1部','负责产品销售','1号楼销售大厅');
--验证触发器执行结果
select from trigger_test --第一条记录为before_insert触发器激活后插入的‘before_insert’,第二条记录为after_insert触发器激活后插入的‘after_insert’,ok
/*FBI warning*/
/*触发器中不能包含事物处理的关键词如:start transaction,commit,rollback,也不能包含call语句*/
/*在触发器执行过程中,任何步骤出错都会阻止程序继续向下执行,但对产生触发事件的普通表来说,已经insert/delete/update过的记录是不能回滚的,更新过的数据将继续保持在表中*/
---删除触发器
基本形式:
drop trigger 触发器名
drop trigger 数据库实例名.触发器名 --触发器是属于数据库实例全局的
例:
drop trigger dept_trig1
--查询是否还存在
select from information_schema.triggers where trigger_name='dept_trig' --查询无结果,Ok
/*FBI warning*/
/*当不再需要某个触发器时,一定要将这个触发器删除,否则会造成数据发生意料之外的变化*/