欢迎分享,转载请注明来源:内存溢出
SQL中,触发器是什么?
1. 基本概念\x0d\x0a触发器是特殊的存储过程,基于一个表创建,主要作用就是实现由主键和外键所不能保证的复杂的参照完整性和数据一致性。\x0d\x0a当触发器所保护的数据发生变化(update,insert,delete)后,自动运行以保证数据的完整性和正确性。通俗的说:通过一个动作(update,insert,delete)调用一个存储过程(触发器)。\x0d\x0a2. 类型\x0d\x0a(1)DML触发器\x0d\x0a在数据库中发生数据 *** 作语言(DML)事件时将启用。DML 事件包括在指定表或视图中修改数据的 INSERT 语句、UPDATE 语句或 DELETE 语句。DML 触发器可以查询其他表,还可以包含复杂的 T-SQL 语句。系统将触发器和触发它的语句作为可在触发器内回滚的单个事务对待,如果检测到错误(例如,磁盘空间不足),则整个事务即自动回滚。\x0d\x0a(2)DDL 触发器\x0d\x0aSQL Server 2005 的新增功能。当服务器或数据库中发生数据定义语言(DDL)事件时将调用这些触发器。但与DML触发器不同的是,它们不会为响应针对表或视图的UPDATE、INSERT或DELETE语句而激发,相反,它们会为响应多种数据定义语言(DDL)语句而激发。这些语句主要是以CREATE、ALTER和DROP开头的语句。DDL触发器可用于管理任务,例如审核和控制数据库 *** 作。\x0d\x0a 创建DML触发器\x0d\x0a1. 使用存储过程模板创建存储过程\x0d\x0a在【对象资源管理器】窗口中,展开“数据库”节点,再展开所选择的具体数据库节点,再展开“表”节点,右击要创建触发器的“表”,选择“新建触发器”命令,如图所示:\x0d\x0a在右侧查询编辑器中出现触发器设计模板,用户可以在此基础上编辑触发器,单击“执行”按钮,即可创建该触发器。\x0d\x0a\x0d\x0a2. 使用T-SQL语句创建表\x0d\x0aCREATE TRIGGER 触发器\x0d\x0aON 表名\x0d\x0aFOR[update,insert,delete ]\x0d\x0aAS SQL语句\x0d\x0a例9-6:创建基于表reader ,DELETE *** 作的触发器。\x0d\x0aUSE Library\x0d\x0aGO\x0d\x0aIF EXISTS(SELECT name FROM sysobjects\x0d\x0a WHERE name='reader_d' AND type='TR')\x0d\x0aDROP TRIGGER reader_d --如果已经存在触发器reader_d则删除\x0d\x0aGO\x0d\x0aCREATE TRIGGER reader_d --创建触发器\x0d\x0aON reader --基于表\x0d\x0aFOR DELETE --删除事件\x0d\x0aAS\x0d\x0aPRINT '数据被删除!'--执行显示输出\x0d\x0aGO\x0d\x0a 试试吧!\x0d\x0a应用:\x0d\x0aUSE Library\x0d\x0aGO\x0d\x0aDELETE reader\x0d\x0awhere Rname='aaa'\x0d\x0a执行结果:\x0d\x0a数据被删除!\x0d\x0a(所影响的行数为 1 行)\x0d\x0a例9-7:在表borrow中添加借阅信息记录时,得到该书的应还日期。\x0d\x0a说明:在表borrow中增加一个应还日期SReturnDate。\x0d\x0aUSE Library\x0d\x0aIF EXISTS (SELECT name FROM sysobjects\x0d\x0aWHERE name ='T_return_date' AND type='TR')\x0d\x0aDROP TRIGGER T_return_date\x0d\x0aGO\x0d\x0aCREATE TRIGGER T_return_date --创建触发器\x0d\x0aON Borrow --基于表borrow\x0d\x0aAfter INSERT --插入 *** 作\x0d\x0aAS\x0d\x0a--查询插入记录INSERTED中读者的类型\x0d\x0aDECLARE @type int,@dzbh char(10),@tsbh char(15)\x0d\x0aSET @dzbh=(SELECT RID FROM inserted)\x0d\x0aSET @tsbh=(SELECT BID FROM inserted)\x0d\x0aSELECT @type= TypeID\x0d\x0aFROM reader\x0d\x0aWHERE RID=(SELECT RID FROM inserted)--副本\x0d\x0a/*把Borrow表中的应还日期改为\x0d\x0a当前日期加上各类读者的借阅期限*/\x0d\x0aUPDATE Borrow SET SReturnDate=getdate()+\x0d\x0aCASE \x0d\x0a WHEN @type=1 THEN 90\x0d\x0a WHEN @type=2 THEN 60\x0d\x0a WHEN @type=3 THEN 30\x0d\x0aEND\x0d\x0aWHERE RID=@dzbh and BID=@tsbh\x0d\x0a应用:\x0d\x0aUSE Library\x0d\x0aINSERT INTO borrow(RID,BID) values('2000186010','TP85-08')\x0d\x0a查看记录:\x0d\x0a \x0d\x0a例9-8:在数据库Library中,当读者还书时,实际上要修改表brorrowinf中相应记录还期列的值,请计算出是否过期。\x0d\x0aUSE Library\x0d\x0aIF EXISTS(SELECT name FROM sysobjects\x0d\x0aWHERE name='T_fine_js' AND type='TR')\x0d\x0aDROP TRIGGER T_fine_js\x0d\x0aGO\x0d\x0aCREATE TRIGGER T_fine_js\x0d\x0aON borrow\x0d\x0aAfter UPDATE\x0d\x0aAS\x0d\x0aDECLARE @days int,@dzbh char(10),@tsbh char(15)\x0d\x0aSET @dzbh=(select RID from inserted)\x0d\x0aSET @tsbh=(select BID from inserted)\x0d\x0aSELECT @days=DATEDIFF(day, ReturnDate, SReturnDate)\x0d\x0a--DATEDIFF函数返回两个日期之差,单位为DAY\x0d\x0aFROM borrow\x0d\x0aWHERE RID=@dzbh and BID=@tsbh\x0d\x0aIF @days>0\x0d\x0a PRINT '没有过期!'\x0d\x0aELSE\x0d\x0a PRINT '过期'+convert(char(6),@days)+'天'\x0d\x0aGO\x0d\x0a应用:\x0d\x0aUSE Library\x0d\x0aUPDATE borrow SET ReturnDate='2007-12-12'\x0d\x0aWHERE RID='2000186010' and BID='TP85-08'\x0d\x0aGO\x0d\x0a执行结果:\x0d\x0a过期-157 天\x0d\x0a(1 行受影响)\x0d\x0a例9-9:对Library库中Reader表的 DELETE *** 作定义触发器。\x0d\x0aUSE Library\x0d\x0aGO\x0d\x0aIF EXISTS(SELECT name FROM sysobjects\x0d\x0a WHERE name='reader_d' AND type='TR')\x0d\x0aDROP TRIGGER reader_d\x0d\x0aGO\x0d\x0aCREATE TRIGGER reader_d\x0d\x0aON Reader\x0d\x0aFOR DELETE\x0d\x0aAS\x0d\x0aDECLARE @data_yj int\x0d\x0aSELECT @data_yj=Lendnum\x0d\x0aFROM deleted\x0d\x0aIF @data_yj>0\x0d\x0a BEGIN\x0d\x0a PRINT '该读者不能删除!还有'+convert(char(2),@data_yj)+'本书没还。\x0d\x0a ROLLBACK\x0d\x0a END\x0d\x0aELSE\x0d\x0a PRINT '该读者已被删除!'\x0d\x0aGO\x0d\x0a应用:\x0d\x0aUSE Library\x0d\x0aGO\x0d\x0aDELETE Reader WHERE RID='2005216119'\x0d\x0a执行结果:\x0d\x0a该读者不能删除!还有4 本书没还。\x0d\x0a 修改触发器\x0d\x0aALTER TRIGGER 触发器\x0d\x0a 删除触发器\x0d\x0aDROP TRIGGER 触发器\x0d\x0a 查看触发器\x0d\x0asp_helptext trigger_name\x0d\x0asp_helptrigger table_name触发器(trigger)是SQL server 提供给程序员和数据分析员来保证数据完整性的一种方法,它是与表事件相关的特殊的存储过程,它的执行不是由程序调用,也不是手工启动,而是由事件来触发,比如当对一个表进行 *** 作( insert,delete, update)时就会激活它执行。触发器经常用于加强数据的完整性约束和业务规则等。 触发器可以从 DBA_TRIGGERS ,USER_TRIGGERS 数据字典中查到。SQL3的触发器是一个能由系统自动执行对数据库修改的语句。\x0d\x0a触发器与存储过程的唯一区别是触发器不能执行EXECUTE语句调用,而是在用户执行Transact-SQL语句时自动触发执行。\x0d\x0a触发器可以查询其他表,而且可以包含复杂的SQL语句。它们主要用于强制服从复杂的业务规则或要求。例如:您可以根据客户当前的帐户状态,控制是否允许插入新订单。\x0d\x0a触发器也可用于强制引用完整性,以便在多个表中添加、更新或删除行时,保留在这些表之间所定义的关系。然而,强制引用完整性的最好方法是在相关表中定义主键和外键约束。如果使用数据库关系图,则可以在表之间创建关系以自动创建外键约束。\x0d\x0aSQL Server 包括三种常规类型的触发器:DML 触发器、DDL 触发器和登录触发器。\x0d\x0a语法为:\x0d\x0aCREATE TRIGGER ``.``\x0d\x0a \x0d\x0aON [dbo] //dbo代表该表的所有者\x0d\x0aFOR EACH ROW\x0d\x0aBEGIN\x0d\x0a --do something\x0d\x0aEND\x0d\x0a 触发器可通过数据库中的相关表实现级联更改,不过,通过级联引用完整性约束可以更有效地执行这些更改。触发器可以强制比用CHECK约束定义的约束更为复杂的约束。与 CHECK 约束不同,触发器可以引用其它表中的列。例如,触发器可以使用另一个表中的 SELECT 比较插入或更新的数据,以及执行其它 *** 作,如修改数据或显示用户定义错误信息。触发器也可以评估数据修改前后的表状态,并根据其差异采取对策。一个表中的多个同类触发器(INSERT、UPDATE 或 DELETE)允许采取多个不同的对策以响应同一个修改语句。\x0d\x0a约束和触发器在特殊情况下各有优势。触发器的主要好处在于它们可以包含使用 Transact-SQL 代码的复杂处理逻辑。因此,触发器可以支持约束的所有功能;但它在所给出的功能上并不总是最好的方法。实体完整性总应在最低级别上通过索引进行强制,这些索引或是 PRIMARY KEY 和 UNIQUE 约束的一部分,或是在约束之外独立创建的。假设功能可以满足应用程序的功能需求,域完整性应通过 CHECK 约束进行强制,而引用完整性(RI) 则应通过 FOREIGN KEY 约束进行强制。在约束所支持的功能无法满足应用程序的功能要求时,触发器就极为有用。\x0d\x0aCHECK 约束只能根据逻辑表达式或同一表中的另一列来验证列值。如果应用程序要求根据另一个表中的列验证列值,则必须使用触发器。约束只能通过标准的系统错误信息传递错误信息。如果应用程序要求使用(或能从中获益)自定义信息和较为复杂的错误处理,则必须使用触发器。\x0d\x0a 触发器可通过数据库中的相关表实现级联更改;不过,通过级联引用完整性约束可以更有效地执行这些更改。触发器可以禁止或回滚违反引用完整性的更改,从而取消所尝试的数据修改。当更改外键且新值与主键不匹配时,此类触发器就可能发生作用。例如,可以在 titleauthor.title_id 上创建一个插入触发器,使它在新值与 titles.title_id 中的某个值不匹配时回滚一个插入。不过,通常使用 FOREIGN KEY 来达到这个目的。\x0d\x0a 如果触发器表上存在约束,则在 INSTEAD OF 触发器执行后但在 AFTER 触发器执行前检查这些约束。如果约束破坏,则回滚 INSTEAD OF 触发器 *** 作并且不执行 AFTER 触发器。\x0d\x0a实例1:insert触发器\x0d\x0acreate trigger tri_insert\x0d\x0aon student\x0d\x0afor insert\x0d\x0aas\x0d\x0adeclare @student_id char(10)\x0d\x0aselect @student_id=s.student_id from\x0d\x0astudent s inner join inserted i\x0d\x0aon s.student_id=i.student_id\x0d\x0aif @student_id='0000000001'\x0d\x0abegin\x0d\x0araiserror('不能插入1的学号!',16,8)\x0d\x0arollback tran\x0d\x0aend\x0d\x0ago\x0d\x0a实例2:update触发器\x0d\x0acreate trigger tri_update\x0d\x0aon student\x0d\x0afor update\x0d\x0aas\x0d\x0aif update(student_id)\x0d\x0abegin\x0d\x0araiserror('学号不能修改!',16,8)\x0d\x0arollback tran\x0d\x0aend\x0d\x0ago\x0d\x0a实例3:delete触发器示\x0d\x0acreate trigger tri_delete\x0d\x0aon student\x0d\x0afor delete\x0d\x0aas\x0d\x0adeclare @student_id varchar(10)\x0d\x0aselect @student_id=student_id from deleted\x0d\x0aif @student_id='admin'\x0d\x0abegin\x0d\x0araiserror('错误',16,8)\x0d\x0arollback tran\x0d\x0aend
赞
(0)
打赏
微信扫一扫
支付宝扫一扫
如何进行存储系统的性能测试?
上一篇
2023-05-03
数据库中的“触发器”是什么意思?
下一篇
2023-05-03
评论列表(0条)