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() ...
随机推荐
- UOJ268 [清华集训2016] 数据交互 【动态DP】【堆】【树链剖分】【线段树】
题目分析: 不难发现可以用动态DP做. 题目相当于是要我求一条路径,所有与路径有交的链的代价加入进去,要求代价最大. 我们把链的代价分成两个部分:一部分将代价加入$LCA$之中,用$g$数组保存:另一 ...
- sql里的正则表达式
SQL语句还可以搭配正则表达式作为查询条件,很是有用. REGEXP_LIKE(匹配)REGEXP_INSTR (包含)REGEXP_REPLACE(替换)REGEXP_SUBSTR(提取) 表 1: ...
- c# Redis 使用
1.服务端两个版本窗口版与安装windows服务版 1.1.窗口版 下载地址:https://github.com/dmajkic/redis/downloads redis-server.exe:服 ...
- MT【287】余弦的线性组合
(2017北大特优)在$\Delta ABC$中,$cos A+\sqrt{2}cos B+\sqrt{2}cos C$的最大值____ 解答 :2$cos A+\sqrt{2}cos B+\sqrt ...
- 【刷题】BZOJ 2759 一个动态树好题
Description 有N个未知数x[1..n]和N个等式组成的同余方程组: x[i]=k[i]*x[p[i]]+b[i] mod 10007 其中,k[i],b[i],x[i]∈[0,10007) ...
- Hdoj 1392.Surround the Trees 题解
Problem Description There are a lot of trees in an area. A peasant wants to buy a rope to surround a ...
- 【CF833E】Caramel Clouds(线段树)
[CF833E]Caramel Clouds(线段树) 题面 CF 洛谷 题解 首先把区间一段一段分出来,那么只有四种情况. 要么没有被任何一朵云被覆盖,那么直接就会产生这一段的贡献. 要么被一朵云覆 ...
- 【php】php目录路径函数系列
在写框架和项目时候我们经常要获取绝对路径,php有内置函数realpath(), 也可以写个函数来实现这个功能 function getAbsolutePath($path) { $path = s ...
- mysql 单表卡死
由于单表数据量过大导致的更新操作处于卡死状态,无法打开也无法修改. 此时需要命令行模式连接数据库,注意点:此处连接需要相同的账号 1. $ SHOW PROCESSLIST; 2. $ kill 37 ...
- js 判断 是否在当前页面 当前页面是否在前端
1.使用visibilitychange 浏览器标签页被隐藏或显示的时候会触发visibilitychange事件. document.addEventListener("visibilit ...