SQLServer触发器实现级联删除

5星 · 超过95%的资源 2 下载量 80 浏览量 更新于2024-09-03 收藏 82KB PDF 举报
"本教程将深入讲解SQL Server中的触发器,特别是如何利用触发器实现级联删除操作。在数据库管理中,级联删除是指当删除一个表中的记录时,与其关联的其他表中的相关记录也会被自动删除。在这个示例中,我们将关注三个表:Navtion_TopSubject(主类别表)、Navtion_NodeSubject(次类别表)和tText(内容表),它们通过外键关联。触发器的作用是在删除主类别表记录时,同时删除次类别表和内容表中与之相关的所有记录,确保数据的一致性。" 在SQL Server中,触发器是一种特殊的存储过程,它会在特定的DML操作(INSERT、UPDATE或DELETE)发生时自动执行。在这个例子中,我们创建了一个名为`RemoveTopAndNodeText`的删除触发器,该触发器在对`Navtion_TopSubject`表执行删除操作时被激活,而不是执行原生的DELETE语句。 触发器的定义始于`CREATE TRIGGER`语句,这里指定了触发器的名字`RemoveTopAndNodeText`,以及它将在`ON [dbo].[Navtion_TopSubject]`表上响应`INSTEAD OF DELETE`事件。`INSTEAD OF`关键字意味着触发器将替代标准的删除行为,而非在其之后执行。 在触发器的主体部分,首先声明了几个变量,如`@fTopID`、`@fNodeCount`、`@fTextCount`和`@fTopName`,用于存储待删除的主类别ID、子类别数量和内容数量。然后,触发器会获取待删除记录的主键值,并开始一个事务来处理这些操作,以确保数据一致性。 在事务中,首先设置一个保存点`SaveTran my_Save1`,以便在出现错误时可以回滚。接着,触发器检查`Navtion_NodeSubject`表中是否有与待删除主类别ID相关的记录。如果存在,再进一步检查`tText`表中是否有相关的内容。如果两者都存在,触发器将先删除`tText`表中的内容,再删除`Navtion_NodeSubject`表中的子类别记录,最后删除`Navtion_TopSubject`表中的主类别记录。如果`tText`表中没有相关记录,那么只需删除子类别记录和主类别记录。 这种级联删除的实现方式确保了数据库的完整性,避免了因删除主类别而遗留孤儿记录的情况。在实际应用中,触发器是维护数据库关系约束和业务规则的重要工具,尤其是在复杂的数据库设计中。然而,使用触发器需谨慎,因为它们可能会影响性能,并可能导致难以预料的副作用。因此,在设计数据库时,应综合考虑触发器和其他约束(如外键约束的级联选项)的使用。
2011-09-01 上传
sqlserver触发器例子 一﹕ 触发器是一种特殊的存储过程﹐它不能被显式地调用﹐而是在往表中插入记录﹑更新记录或者删除记录时被自动地激活。所以触发器可以用来实现对表实施复杂的完整性约`束。 二﹕ SQL Server为每个触发器都创建了两个专用表﹕Inserted表和Deleted表。这两个表由系统来维护﹐它们存在于内存中而不是在数据库中。这两个表的结构总是与被该触发器作用的表的结构相同。触发器执行 完成后﹐与该触发器相关的这两个表也被删除。 Deleted表存放由于执行Delete或Update语句而要从表中删除的所有行。 Inserted表存放由于执行Insert或Update语句而要向表中插入的所有行。 三﹕Instead of 和 After触发器 SQL Server2000提供了两种触发器﹕Instead of 和After 触发器。这两种触发器的差别在于他们被激活的同﹕ Instead of触发器用于替代引起触发器执行的T-SQL语句。除表之外﹐Instead of 触发器也可以用于视图﹐用来扩展视图可以支持的更新操作。 After触发器在一个Insert,Update或Deleted语句之后执行﹐进行约束检查等动作都在After触发器被激活之前发生。After触发器只能用于表。 一个表或视图的每一个修改动作(insert,update和delete)都可以有一个instead of 触发器﹐一个表的每个修改动作都可以有多个After触发器。 四﹕触发器的执行过程 如果一个Insert﹑update或者delete语句违反了约束﹐那幺After触发器不会执行﹐因为对约束的检查是在After触发器被激动之前发生的。所以After触发器不能超越约束。 Instead of 触发器可以取代激发它的操作来执行。它在Inserted表和Deleted表刚刚建立﹐其它任何操作还没有发生时被执行。因为Instead of 触发器在约束之前执行﹐所以它可以对约束进行一些预处理。 五﹕使用T-SQL语句来创建触发器 基本语句如下﹕ create trigger trigger_name on {table_name | view_name} {for | After | Instead of } [ insert, update,delete ] as sql_statement 六﹕相关示例﹕ 1﹕在Orders表中建立触发器﹐当向Orders表中插入一条订单记录时﹐检查goods表的货品状态status是否为1(正在整理)﹐是﹐则不能往Orders表加入该订单。 create trigger orderinsert on orders after insert as if (select status from goods,inserted where goods.name=inserted.goodsname)=1 begin print 'the goods is being processed' print 'the order cannot be committed' rollback transaction --回滚﹐避免加入 end 2﹕在Orders表建立一个插入触发器﹐在添加一条订单时﹐减少Goods表相应的货品记录中的库存。 create trigger orderinsert1 on orders after insert as update goods set storage=storage-inserted.quantity from goods,inserted where goods.name=inserted.goodsname 3﹕在Goods表建立删除触发器﹐实现Goods表和Orders表的级联删除。 create trigger goodsdelete on goods after delete as delete from orders where goodsname in (select name from deleted) 4﹕在Orders表建立一个更新触发器﹐监视Orders表的订单日期(OrderDate)列﹐使其不能手工修改. create trigger orderdateupdate on orders after update as if update(orderdate) begin raiserror(' orderdate cannot be modified',10,1) rollback transaction end 5﹕在Orders表建立一个插入触发器﹐保证向Orders表插入的货品名必须要在Goods表中一定存在。 create trigger orderinsert3 on orders after insert as if (select count(*) from goods,inserted where goods.name=inserted.goodsname)=0 begin print ' no entry in goods for this order' rollback transaction end --insert 触发器 create trigger tri_infoDetails_i on info_details after insert as declare @id int begin --delete from info_details where id= select @id=id from inserted; insert into info_details_index(TYPE,TITLE,content,POST_TIME,FLAG) select type,title,content,getdate(),1 from info_details where id=@id; --update info_details_index set content=content end; -- update触发器 --select top 0 type,title,content,getdate() as post_time,1 as flag into info_details_index from info_details; create trigger tri_infoDetails_u on info_details after update as declare @id int begin if exists(select 1 from inserted) if exists(select 1 from deleted) begin select @id=id from inserted; insert into info_details_index(TYPE,TITLE,content,POST_TIME,FLAG)select type,title,content,getdate(),-1 from info_details where id=@id; insert into info_details_index(TYPE,TITLE,content,POST_TIME,FLAG)select type,title,content,getdate(),1 from info_details where id=@id; end --update info_details_index set content=content end --delete触发器 create trigger tri_infoDetails_d on info_details after delete as declare @id int begin if exists(select 1 from deleted) begin insert into info_details_index(TYPE,TITLE, POST_TIME,FLAG) select type,title, getdate(),-1 from deleted info_details ; end end
2010-05-07 上传
定义: 何为触发器?在SQL Server里面也就是对某一个表的一定的操作,触发某种条件,从而执行的一段程序。触发器是一个特殊的存储过程。 常见的触发器有三种:分别应用于Insert , Update , Delete 事件。 我为什么要使用触发器?比如,这么两个表: Create Table Student( --学生表 StudentID int primary key, --学号 .... ) Create Table BorrowRecord( -学生借书记录表 BorrowRecord int identity(1,1), --流水号 StudentID int , --学号 BorrowDate datetime, --借出时间 ReturnDAte Datetime, --归还时间 ... ) 用到的功能有: 1.如果我更改了学生的学号,我希望他的借书记录仍然与这个学生相关(也就是同时更改借书记录表的学号); 2.如果该学生已经毕业,我希望删除他的学号的同时,也删除它的借书记录。 这时候可以用到触发器。对于1,创建一个Update触发器: Create Trigger truStudent On Student --在Student表中创建触发器 for Update --为什么事件触发 As --事件触发后所要做的事情 if Update(StudentID) begin Update BorrowRecord Set StudentID=i.StudentID From BorrowRecord br , Deleted d ,Inserted i --Deleted和Inserted临时表 Where br.StudentID=d.StudentID end 理解触发器里面的两个临时的表:Deleted , Inserted 。注意Deleted 与Inserted分别表示触发事件的表“旧的一条记录”和“新的一条记录”。 一个Update 的过程可以看作为:生成新的记录到Inserted表,复制旧的记录到Deleted表,然后删除Student记录并写入新纪录。