设为首页 收藏本站
查看: 714|回复: 0

[经验分享] 浅谈SQL Server触发器的使用

[复制链接]

尚未签到

发表于 2016-11-2 00:47:02 | 显示全部楼层 |阅读模式

浅谈SQL Server触发器的使用

收藏


  来源:http://blog.csdn.net/zouhu562/archive/2009/08/01/4400812.aspx

  
  
<script type="text/javascript">
document.body.oncopy = function() {
if (window.clipboardData) {
setTimeout(function() {
var text = clipboardData.getData(&quot;text&quot;);
if (text &amp;&amp; text.length&gt;300) {
text = text + &quot;\r\n\n本文来自CSDN博客,转载请标明出处:&quot; + location.href;
clipboardData.setData(&quot;text&quot;, text);
}
}, 100);
}
}
</script><script type="text/javascript">function StorePage(){d=document;t=d.selection?(d.selection.type!='None'?d.selection.createRange().text:''):(d.getSelection?d.getSelection():'');void(keyit=window.open('http://www.365key.com/storeit.aspx?t='+escape(d.title)+'&amp;u='+escape(d.location.href)+'&amp;c='+escape(t),'keyit','scrollbars=no,width=475,height=575,left=75,top=20,status=no,resizable=yes'));keyit.focus();}</script>
  http://www.cftea.com/specials/trigger/



  
  摘要:

触发器(trigger)是个特殊的存储过程,它的执行不是由程序调用,也不是手工启动,而是由个事件来触发。本文将介绍SQL Server触发器的使用
  触发器建立的代码
  


Create Trigger TG_ProjectName
On table1
After Update
As
Update table2
Set [工程名]=b.工程名
from table2 a,inserted b
where a.ProjID = b.ID
  关于触发器中Inserted和Deleted的解释。

  inserted触发器语句中使用了两种特殊的表:deleted 表和 inserted 表。Microsoft® SQL Server
2000
自动创建和管理这些表。可以使用这两个临时的驻留内存的表测试某些数据修改的效果及设置触发器操作的条件;然而,不能直接对表中的数据进行更改。
  inserted 和 deleted 表主要用于触发器中:

  ◆扩展表间引用完整性。
  ◆在以视图为基础的基表中插入或更新数据。
  ◆检查错误并基于错误采取行动。
  ◆找到数据修改前后表状态的差异,并基于此差异采取行动。
  Deleted 表用于存储 DELETE 和 UPDATE 语句所影响的行的复本。在执行 DELETE 或 UPDATE 语句时,行从触发器表中删除,并传输到 deleted 表中。Deleted 表和触发器表通常没有相同的行。
  Inserted 表用于存储 INSERT 和 UPDATE 语句所影响的行的副本。在一个插入或更新事务处理中,新建行被同时添加到 inserted 表和触发器表中。Inserted 表中的行是触发器表中新行的副本。
  更新事务类似于在删除之后执行插入;首先旧行被复制到 deleted 表中,然后新行被复制到触发器表和 inserted 表中。
  在设置触发器条件时,应当为引发触发器的操作恰当使用 inserted 和 deleted 表。虽然在测试 INSERT 时引用
deleted 表或在测试 DELETE 时引用 inserted 表不会引起任何错误,但是在这种情形下这些触发器测试表中不会包含任何行。
  说明

  如果触发器操作取决于一个数据修改所影响的行数,应该为多行数据修改(基于 SELECT 语句的 INSERT、DELETE 或 UPDATE)使用测试(如检查 @@ROWCOUNT),然后采取相应的对策。
  SQL Server 2000不允许AFTER 触发器引用 inserted 和 deleted 表中的 text、ntext 或 image 列;然而,允许 INSTEAD OF 触发器引用这些列。有关更多信息,请参见 CREATE TRIGGER。
  在 INSTEAD OF 触发器中使用 inserted 和 deleted 表

  传递到在表上定义的 INSTEAD OF 触发器的 inserted 和 deleted 表遵从与传递到 AFTER 触发器的
inserted 和 deleted 表相同的规则。inserted 和 deleted 表的格式与在其上定义 INSTEAD OF
触发器的表的格式相同。inserted 和 deleted 表中的每一列都直接映射到基表中的列。
  有关引用带 INSTEAD OF 触发器的表的 INSERT 或 UPDATE 语句何时必须提供列值的规则与表没有 INSTEAD OF 触发器时相同:
  不能为计算列或具有 timestamp 数据类型的列指定值。
  不能为具有 IDENTITY 属性的列指定值,除非该列的 IDENTITY_INSERT 为 ON。当 IDENTITY_INSERT
为 ON 时,INSERT 语句必须提供一个值。 INSERT 语句必须为所有无 DEFAULT 约束的 NOT NULL 列提供值。

对于除计算列、标识列或 timestamp 列以外的任何列,任何允许空值的列或具有 DEFAULT 定义的 NOT NULL 列的值都是可选的。
  当 INSERT、UPDATE 或 DELETE 语句引用具有 INSTEAD OF
触发器的视图时,数据库引擎将调用该触发器,而不是对任何表采取任何直接操作。即使为视图生成的 inserted 和 deleted
表中的信息格式与基表中的数据格式不同,该触发器在生成执行基表中的请求操作所需的任何语句时,仍必须使用 inserted 和 deleted
表中的信息。

传递到在视图上定义的 INSTEAD OF 触发器的 inserted 和 deleted 表格式与为该视图定义的 SELECT 语句的选择列表相匹配。例如:
  


CREATE VIEW EmployeeNames (EmployeeID, LName, FName)
AS
SELECT EmployeeID, LastName, FirstName
FROM Northwind.dbo.Employees
  视图的结果集有三列:一个 int 列和两个 nvarchar 列。传递到在视图上定义的 INSTEAD OF 触发器的 inserted
和 deleted 表也具有名为 EmployeeID 的 int 列、名为 LName 的 nvarchar 列和名为 FName 的
nvarchar 列。
  视图的选择列表还包含不直接映射到单个基表列的表达式。一些视图表达式(如常量调用或函数调用)可能不引用任何列,这类表达式会被忽略。复杂的表达
式会引用多列,但在 inserted 和 deleted
表中,每个插入的行仅有一个值。如果视图中的简单表达式引用具有复杂表达式的计算列,则这些简单表达式也有同样的问题。视图上的 INSTEAD OF
触发器必须处理这些类型的表达式。有关更多信息,请参见视图上 INSTEAD OF 触发器中的表达式和计算列。
  顺便说一下,当对某张表建立触发器后,分3种情况讨论

  1.插入操作(Insert)
  Inserted表有数据,Deleted表无数据
  2.删除操作(Delete)
  Inserted表无数据,Deleted表有数据
  3.更新操作(Update)
  Inserted表有数据(新数据),Deleted表有数据(旧数据)
  本章描述如何书写触发器函数。 触发器函数可以用 C 或者任何其它可用的过程语言编写。 目前不可能用 SQL 语言书写触发器。


32.1. 触发器行为概述

  一个触发器函数可以再一个INSERT
,UPDATE
, 或者 DELETE
命令之前或者之后执行,要么是对每个被修改的行一次, 要么是每条 SQL
一次。 如果发生触发器事件,那么将在合适的时刻调用触发器的函数以处理该事件。
  触发器函数必须在创建触发器之前,作为一个没有参数并且返回trigger
类型的函数定义。 (触发器函数通过特殊的 TriggerData
结构接收其输入,而不是用普通函数参数那种形式。)
  一旦创建了一个合适的触发器函数,触发器就用 CREATE TRIGGER
创建。同一个触发器函数可以用于多个触发器。
  有两种类型的触发器:按行触发的触发器和按语句触发的触发器。在按行触发的触发器里,
触发器函数是为触发触发器的语句影响的每一行执行一次。相比之下,一个按语句触发的触发器是在每执行一次合适的语句执行一次的,
而不管影响的行数。特别是,一个影响零行的语句将仍然导致任何适用的按语句触发的触发器的执行。 这两种类型的触发器有时候分别叫做"行级别的触发器"
和"语句级别的触发器"

  语句级别的 "before"
触发器通常在语句开始做任何事情之前触发, 而语句级别的 "after"
触发器在语句的最后触发。 行级别的 "before"
触发器在对特定行进行操作的时候马上触发, 而行级别的 "after"
触发器在语句结束的时候触发(但是在任何语句级别的 "after"
触发器之前)。
  按语句触发的触发器应该总是返回 NULL
。 如果必要,按行触发的触发器函数可以给调用它的执行者返回一表数据行(一个类型为 HeapTuple
的数值), 那些在操作之前触发的触发器有以下选择:


  •   它可以返回 NULL
    以忽略对当前行的操作。 这就指示执行器不要执行调用该触发器的行级别操作(对特定行的插入或者更改))。

  •   只用于INSERT
    和UPDATE
    触发器: 返回的行将成为被插入的行或者是成为将要更新的行。 这样就允许触发器函数修改被插入或者更新的行。

  一个无意导致任何这类行为的在操作之前触发的行级触发器必须仔细返回那个被当作新行传进来的同一行 (也就是说,对于 INSERT
和 UPDATE
触发器而言,是 NEW
行, 对于 DELETE
触发器而言,是 OLD
行)。
  对于在操作之后触发的行级别的触发器,其返回值会被忽略,因此他们可以返回NULL

  如果多于一个触发器为同样的事件定义在同样的关系上, 触发器将按照由名字的字母顺序排序的顺序触发。 如果是事件之前触发的触发器,每个触发器返回的可能已经被修改过的行成为下一个触发器的输入。 如果任何事件之前触发的触发器返回 NULL
指针, 那么其操作被丢弃并且随后的触发器不会被触发。
  通常,行的 before 触发器用于检查或修改将要插入或者更新的数据。 比如,一个 before
触发器可以用于把当前时间插入一个时间戳字段, 或者跟踪该行的两个元素是一致的。行的 after 触发器多数用于填充或者更新其它表,
或者对其它表进行一致性检查。这么区分工作的原因是, after 触发器肯定可以看到该行的最后数值, 而 before
触发器不能;还可能有其它的 before 触发器在其后触发。 如果你没有具体的原因定义触发器是 before 还是 after,那么
before 触发器的效率高些, 因为操作相关的信息不必保存到语句的结尾。
  如果一个触发器函数执行 SQL 命令,然后这些命令可能再次触发触发器。 这就是所谓的级联触发器。对级联触发器的级联深度没有明确的限制。 有可能出现级联触发器导致同一个触发器的递归调用的情况; 比如,一个 INSERT
触发器可能执行一个命令, 把一个额外的行插入同一个表中,导致 INSERT
触发器再次激发。 避免这样的无穷递归的问题是触发器程序员的责任。
  在定义一个触发器的时候,我们可以声明一些参数。
在触发器定义里面包含参数的目的是允许类似需求
的不同触发器调用同一个函数。 比如,我们可能有一个通用的触发器函数, 接受两个字段名字,把当前用户放在第一个,而当前时间戳在第二个。
只要我们写得恰当,那么这个触发器函数就可以和触发它的特定表无关。 这样同一个函数就可以用于有着合适字段的任何表的 INSERT
事件,实现自动跟踪交易表中的记录创建之类的问题。如果定义成一个 UPDATE
触发器,我们还可以用它跟踪最后更新的事件。
  每种支持触发器的编程语言都有自己的方法让触发器函数得到输入数据。 这些输入数据包括触发器事件的类型(比如,INSERT
或者 UPDATE
)以及所有在 CREATE TRIGGER
里面列出的参数。 对于低层次的触发器,输入数据也包括 INSERT
和 UPDATE
触发器的 NEW
行,和/或 UPDATE
和 DELETE
触发器的 OLD
行。 语句级别的触发器目前没有任何方法检查改语句修改的独立行。

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-294252-1-1.html 上篇帖子: SQL Server教程:详细学习游标 下篇帖子: SQL SERVER性能优化综述(转)
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表