Sqlserver中的触发器
一 什么是触发器
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:表示可以删除多个触发器的占位符
参考文章:
1、http://www.cnblogs.com/wangprince2017/p/7827091.html
2、http://www.cnblogs.com/wangprince2017/p/7827091.html
Sqlserver中的触发器的更多相关文章
- SqlServer 中的触发器
SqlServer 触发器实现多表之间同步增加.删除与更新 定义: 何为触发器?在SQL Server里面也就是对某一个表的一定的操作,触发某种条件,从而执行的一段程序.触发器是一个特殊的存储过程. ...
- 知方可补不足~sqlserver中触发器的使用
回到目录 触发器在过去的10年中,即存储过程和ado.net称霸江湖期间是那么的重要,而现在,trigger显得不是那么必要的,我们很少将复杂的业务写在SQL里,当然也会没有机会写到trigger里了 ...
- Sqlserver中存储过程,触发器,自定义函数(二)
Sqlserver中存储过程,触发器,自定义函数: 自定义函数:1.函数类型:2.函数的参数和返回值: 1.函数类型:标量值函数,返回的是一个标量值表值函数:内联表值函数:多语句表值函数. 标量值函数 ...
- Sqlserver中存储过程,触发器,自定义函数(一)
Sqlserver中存储过程,触发器,自定义函数 1.存储过程有关内容存储过程的定义:存储过程的分类:存储过程的创建,修改,执行:存储过程中参数的传递,返回与接收:存储过程的返回值:存储过程使用游标. ...
- Sqlserver中存储过程,触发器,自定义函数
Sqlserver中存储过程,触发器,自定义函数: 1. 触发器:是针对数据库表或数据库的特殊存储过程,在某些行为发生的时候就会被激活 触发器的分类: DML触发器:发生在数据操作语言执行时触发执行的 ...
- C#批量插入数据到Sqlserver中的四种方式
我的新书ASP.NET MVC企业级实战预计明年2月份出版,感谢大家关注! 本篇,我将来讲解一下在Sqlserver中批量插入数据. 先创建一个用来测试的数据库和表,为了让插入数据更快,表中主键采用的 ...
- SQLSERVER中的假脱机spool
SQLSERVER中的假脱机spool 我发现网上对于假脱机的解释都非常零散,究竟假脱机是什么? 这几天在家里研究了一下,收集了很多网上的资料 假脱机是中文的翻译,而英文的名字叫做 spool 在徐老 ...
- SqlServer基础之(触发器)
概念: 触发器(trigger)是SQL server 提供给程序员和数据分析员来保证数据完整性的一种方法,它是与表事件相关的特殊的存储过程,它的执行不是由程序调用,也不是手工启动,而是由事件来触 ...
- (转)笔记320 SQLSERVER中的加密函数 2013-7-11
1 --SQLSERVER中的加密函数 2013-7-11 2 ENCRYPTBYASYMKEY() --非对称密钥 3 ENCRYPTBYCERT() --证书加密 4 ENCRYPTBYKEY() ...
随机推荐
- 怎么用Verilog语言描述同步FIFO和异步FIFO
感谢 知乎龚大佬 打杂大佬 网上几个nice的博客(忘了是哪个了....) 前言 虽然FIFO都有IP可以使用,但理解原理还是自己写一个来得透彻. 什么是FIFO? Fist in first out ...
- 【AGC018F】Two Trees 构造 黑白染色
题目描述 有两棵有根树,顶点的编号都是\(1\)~\(n\). 你要给每个点一个权值\(a_i\),使得对于两棵树的所有顶点\(x\),满足\(|x\)的子树的权值和\(|=1\) \(n\leq 1 ...
- Git回滚代码暴力法
Git回滚有多种方式,这里使用的是[强制提交到远程分支] 效果为:如回滚前的提交记录是 1.2.3.4,使用这种方法回滚到2,那么提交记录就变成了1.2. 操作方法: 需要在本地的Git仓库,右键选择 ...
- Breakable loop in Scratch
Breakable loop in Scratch https://stackoverflow.com/questions/30682144/breakable-loop-in-scratch Bre ...
- Centos7安装OpenDCIM-19.01步骤
Centos7安装OpenDCIM-19.01步骤 openDCIM是一款免费的开源解决方案,用于管理数据中心基础设施.它已经被几家企业组织所使用,由于开发人员的不懈努力,正在迅速完善. openDC ...
- html内嵌框架
html内嵌框架 <iframe>标签会创建包含另外一个html文件的内联框架(即行内框架),src属性来定义另一个html文件的引用地址,frameborder属性定义边框,scroll ...
- [SHOI2001]化工厂装箱员(dp?暴力:暴力)
118号工厂是世界唯一秘密提炼锎的化工厂,由于提炼锎的难度非常高,技术不是十分完善,所以工厂生产的锎成品可能会有3种不同的纯度,A:100%,B:1%,C:0.01%,为了出售方便,必须把不同纯度 ...
- 微信支付退款(PHP后端)
应用场景 当交易发生之后一段时间内,由于买家或者卖家的原因需要退款时,卖家可以通过退款接口将支付款退还给买家,微信支付将在收到退款请求并且验证成功之后,按照退款规则将支付款按原路退到买家帐号上. 微信 ...
- goaccess nginx 日志分析
用法介绍 GoAccess的基本语法如下: goaccess [ -b ][ -s ][ -e IP_ADDRESS][ - a ] <-f log_file > 参数说明: -f – 日 ...
- 洛谷P1463 反素数
经典题了,很难想到这TM是搜索...... 题意:求[1, n]中约数最多的数中最小的. 解:我们有约数个数定理. 所以考虑通过枚举每个质因数个数来直接计算出约数个数. 然后就可以搜索了. 注意:若p ...