SqlServer中自动ID的必要性和触发器中一种隐式错误

本文探讨了在SQL Server中使用触发器时遇到的问题,特别是当触发器处理多条记录时的解决方案。介绍了如何避免使用特定的SELECT语句,并提供了一种更稳定的方法来更新关联表。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

        在SqlServer中自动ID可以说很重要,如果没有它,可能有的功能就难以实现。用select C1 from Deleted 这样的语句做赋值做比较,一般情况下,如果是单独触发,都是得到单个值,这时候是没问题的。但如果多个关系表级联触发或者其它更新语句间接触发更新数据,就会出错,说什么子查询不能用在=、>、<、<>等运算中。因为此时得到的是理论上是多条记录,是一个数据集,自然就不能像单个值那样运算了。为此可以用下面的方法解决。有表Table1,Table2,Table3。Table1触发Table2用T1,Table2触发Table3用T2。
     T1:update Table2 set C1=(select a from Inserted) from Table2 join Deleted on Table2.C1=Deleted.a
     T2--update A set a=(select C1 from Inserted) from Table3 A join Deleted D on A.a=D.C1 and A.b=D.C2
--update A set b=(select C2 from Inserted) from Table3 A join Deleted D on A.a=D.C1 and A.b=D.C2
--if (select C1 from Inserted)<>(select C1 from Deleted)     --问题之所在,如果得到的结果是多条就出错,级联触发就会产生多记录,必定出错。
--如下两种方法都可以,不要用 select C1 from Deleted 这样的
update A set a=I.C1,b=I.C2 from Table3 A join Deleted D on A.a=D.C1 and A.b=D.C2 join Inserted I on D.ID=I.ID and (I.C1<>D.C1 or I.C2<>D.C2)
--select D.C1,D.C2,I.C1 C1_New,I.C2 C2_New into #Edit from Inserted I join Deleted D on I.ID=D.ID and (I.C1<>D.C1 or I.C2<>D.C2)
--if (select count(*) from #Edit)>0
--update A set a=#Edit.C1_New,b=#Edit.C2_New from Table3 A join #Edit on A.a=#Edit.C1 and A.b=#Edit.C2
     以后触发器中就不要用select C1 from Deleted 这种形式的写法了,即使不是级联触发也不用,如果用Update语句还是会引起这种问题。
     与此同时,在上面的写法中都有一个自动增量的唯一的不可修改的ID,用于确定Deleted和Inserted的关联,否则无法关联。 
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值