一 什么是触发器
1.1 触发器的概念
触发器(trigger)是SQL server来保证数据完整性的一种方法,它是与表事件相关的特殊的存储过程,它的执行是由事件来触发,当对一个表进行操作( insert,delete, update)时就会激活它执行。触发器经常用于加强数据的完整性约束和业务规则等。
触发器和存储过程的区别:
触发器与存储过程的区别是运行方式的不同,触发器在执行T-SQL语句时自动触发执行,而存储过程需要用户,应用程序或者触发器来显式地调用并执行。
1.2 触发器的优点和使用场景
1.触发器是自动的。当对表中的数据做了任何修改之后立即被激活。
2.级联修改数据库中所有相关的表,自动触发其它与之相关的操作(删除角色,所有使用该角色的用户设置为默认的低权限角色)
3.触发器可以强制限制。这些限制比用CHECK约束所定义的更复杂。与CHECK约束不同的是,触发器可以引用其他表中的列。
4.返回自定义的错误消息。CHECK约束无法返回信息,而触发器可以
5.触发器可以调用更多的存储过程
二 DML触发器的工作原理
inserted表和deleted表对照:
修改操作记录 | inserted表 | deleted表 |
增加(insert)记录 | 存放新增的记录 | ............ |
删除(deleted)记录 | .............. | 存放被删除的记录 |
修改(update)记录 | 存放更新后的记录 | 存放更新前的记录 |
执行顺序:
三 使用DML触发器
准备测试数据:
--创建用户表
create table UserInfo(
userId int identity(1,1) primary key,
userName nvarchar (20),
userGender char(2),
userRoleId int not null
) --创建角色表
create table RoleInfo(
roleId int identity(1,1) primary key,
roleName nvarchar(20)
)
3.1 创建insert触发器
作用:插入一个新userinfo,统计userinfo个数
--创建insert触发器
create trigger trig_insertUser on userinfo
after insert
as
begin
if object_id(N'userSum',N'U') is null --判断userSum表是否存在
create table userSum (numbers int default(0));
declare @userCount int;
select @userCount=count(*) from userinfo;
if not exists(select * from userinfo) --判断userInfo表是否有数据
insert into userSum (numbers) values(0);
update userSum set numbers=@userCount;--插入新用户后更新统计数据
end --执行
insert into userinfo (username,usergender,userroleid) values('zs','男',1);
select numbers as '用户人数' from userSum
insert into userinfo (username,usergender,userroleid) values('ls','男',1);
select numbers as '用户人数' from userSum
insert into userinfo (username,usergender,userroleid) values('ww','男',2);
select numbers as '用户人数' from userSum
执行结果:
userSum在插入数据时自动更新,应该禁止用户向其中直接插入数据
--创建禁止插入的触发器
create trigger trig_insertForbidden on userSum
after insert
as
begin
RAISERROR('禁止直接插入记录,操作被禁止',1,1);
rollback tran
end --执行
insert into userSum (numbers) values(10)
结果如下:
3.2 创建delete触发器
作用:删除角色,把使用该角色的用户设置为“默认角色”
--删除角色的delete触发器
create trigger trig_deleterole on roleinfo
after delete
as
begin
declare @roleid int;
select @roleid= roleid from deleted;--删除的角色Id
select rolename as 删除的角色 from deleted;--删除的角色
update userinfo set userroleid=1 where userroleid=@roleid --修改使用该角色的用户的角色为默认角色
end --执行
delete from roleinfo where roleid=2
角色表:
删除角色2前用户表: 删除角色2后的用户表 :
3.3 创建UPDATE触发器
update触发器是当用户在指定表上执行update语句时被调用被调用,这种类型的触发器用来约束用户对数据的修改。update触发器可以执行两种操作:更新前的记录存储在deleted表中,更新后的记录存储在inserted表中。
--更新用户的update触发器
create trigger trig_userupdate on userinfo
after update
as
begin
select username as 更新前的用户, usergender as 性别 from deleted --更新前
select username as 更新后的用户, usergender as 性别 from inserted--更新后
end --执行
update userinfo set username='newzs',usergender='女' where userid=23 --注:编号23是一个用户名为'zs',性别'男'的用户
执行结果:
3.4 创建instead of 触发器
作用:执行删除操作时用逻辑删除替代
在删除用户时,我们通常不希望真正去把用户信息删除掉,而是设置一个删除标记,表示该用户已被删除了(逻辑删除)。这里在用户表中添加“isDeleted”字段<bit类型>,false表示未删除,true表示已删除,表结构如下:
创建触发器的代码
--创建instead of触发器
create trigger trig_insteadofdel on userinfo
instead of delete
as
begin
declare @userid int;
select @userid=userid from deleted ;
update userinfo set isdeleted=1 where userid=@userid;
select * from deleted;--逻辑删除前
select * from userinfo where userid=@userid;--逻辑删除后
end --执行
delete userinfo where userid=23
执行结果(可以看出并没有真正去执行删除操作,而是被逻辑删除替代了):
3.5 嵌套触发器介绍
如果一个触发器在执行操作时调用了另外一个触发器,而这个触发器又接着调用了下一个触发器,那么就形成了嵌套触发器。嵌套触发器在安装时就被启用,但是可以使用系统存储过程sp_configure禁用和重新启用嵌套触发器。
嵌套触发器不一定要形成一个环,它可以 T1->T2->T3...这样一直触发下去,最多允许嵌套 32 层。如果嵌套的次数超过限制,那么该触发器将被终止,并回滚整个事务,使用嵌套触发器需要注意以下几点:
- 默认情况下,嵌套触发器配置选项是开启的。
- 在同一个触发器事务中,一个嵌套触发器不能被触发两次。
- 由于触发器是一个事务,如果在一系列嵌套触发器的任意层次中发生错误,则整个事务都将取消,而且所有数据回滚。
嵌套是用来保持整个数据库的完整性的重要功能,但有时可能需要禁用嵌套,如果禁用了嵌套,那么修改一个触发器的实现不会再触发该表上的任何触发器。在下述情况下,需要禁用嵌套触发器:
- 嵌套触发要求复杂而有理论的设计,级联修改可能会修改用户不想涉及的数据。
- 在一系列嵌套触发器中的任意点的时间修改操作都会触发一些触发器,尽管这时数据库提供很强的保护功能,但如果以特定的顺序更新表,就会产生问题。
使用下列语句禁用嵌套和再次启用嵌套:
--禁用嵌套
exce sp_configure 'nested triggers',0;
--启用嵌套
exce sp_configure 'nested triggers',1;
3.6 递归触发器
触发器的递归是指一个触发器从其内部再一次激活该触发器。例如update操作激活的触发器内部还有一条数据表的更新语句,那么这个更新语句就有可能激活这个触发器本身,当然,这种递归的触发器内部还会有判断语句,只有一定情况下才会执行那个T_SQL语句,否则就成为无线调用的死循环了。
SqlServer中的递归触发器包括两种:直接递归和间接递归。
- 直接递归:触发器被触发后并执行一个操作,而该操作又使用一个触发器再次被触发。
- 间接递归:触发器被触发并执行一个操作,而该操作又使另一个表中的某个触发器被触发,第二个触发器使原始表得到更新,从而再次触发第一个触发器。
默认情况下,递归触发器选项是禁用的。递归触发器最多只能递归16层,如果递归中的第16个触发器激活了第17个触发器,则结果与发布的rollback命令一样,所有数据都将回滚。
我们举例解释如下,假如有表1、表2名称分别为 T1、T2,在 T1、T2 上分别有触发器 G1、G2。
- 间接递归:对 T1 操作从而触发 G1,G1 对 T2 操作从而触发 G2,G2 对 T1 操作从而再次触发 G1...
- 直接递归:对 T1 操作从而触发 G1,G1 对 T1 操作从而再次触发 G1...
设置直接递归:
默认情况下是禁止直接递归的,要设置为允许有两种方法:
- T-SQL:exec sp_dboption 'dbName', 'recursive triggers', true;
- EM:数据库上点右键->属性->选项。
四 管理触发器
4.1 查看触发器
1 查看数据库中所有的触发器
--查看数据库中所有的触发器
use 数据库名
go
select * from sysobjects where xtype='TR'
sysobjects 保存着数据库的对象,其中 xtype 为 TR 的记录即为触发器对象。在 name 一列,我们可以看到触发器名称。
2 sp_helptext 查看触发器内容
use 数据库名
go
exec sp_helptext '触发器名称'
将会以表的样式显示触发器内容。
除了触发器外,sp_helptext 还可以显示 规则、默认值、未加密的存储过程、用户定义函数、视图的文本。
3 sp_helptrigger 用于查看触发器的属性
sp_helptrigger 有两个参数:第一个参数为表名;第二个为触发器类型,为 char(6) 类型,可以是 INSERT、UPDATE、DELETE,如果省略则显示指定表中所有类型触发器的属性。
use 数据库名
go
exec sp_helptrigger tableName
4.2 禁用启用触发器
禁用:alter table 表名 disable trigger 触发器名称
启用:alter table 表名 enable trigger 触发器名称
如果有多个触发器,则各个触发器名称之间用英文逗号隔开。
如果把“触发器名称”换成“ALL”,则表示禁用或启用该表的全部触发器。
4.3 修改触发器
--修改触发器语法
ALTER TRIGGER trigger_name
ON table_name
[ WITH ENCRYPTION ]
FOR {[DELETE][,][INSERT][,][UPDATE]}
AS
sql_statement;
4.4 删除触发器
--语法格式:
DROP TRIGGER { trigger } [ ,...n ]
参数:
trigger: 要删除的触发器名称
n:表示可以删除多个触发器的占位符
参考文章: