SQL Server之DML触发器
一、如何创建一个触发器呢
触发器的定义语言如下:
CREATE [ OR ALTER ] TRIGGER trigger_name
on {table_name | view_name}
{for | After | Instead of }
[ insert, update,delete ]
as
sql_statement
从这个定义语言我们可以知道如下信息:
- trigger_name:触发器的名称。 trigger_name 必须遵循标识符规则,但 trigger_name 不得以 # 或 ## 开头。
- table | view:触发器可以作用于表或视图,只有 INSTEAD OF 触发器才能引用视图
- FOR | AFTER:FOR 或 AFTER 指定仅当触发 SQL 语句中指定的所有操作都已成功启动时,DML 触发器才触发。
- INSTEAD OF:指定 DML 触发器(而不是触发 SQL 语句)启动,因此替代触发语句的操作。
从定义中我们可以发现:DML触发器总体有两种类型:AFTER 触发器和INSTEAD OF 触发器
二、AFTER 触发器和INSTEAD OF 触发器的区别
直接区别它们可能晦涩难懂,我们先举个例子
(1)建一张表
--作家表
CREATE TABLE my_test.dbo.author (
id bigint IDENTITY(0,1) NOT NULL,
name varchar(100) COLLATE SQL_Latin1_General_CP1_CI_AS NULL )
(2)在author创建after触发器
CREATE TRIGGER authorTrigger ON author AFTER INSERT AS
declare @count int
BEGIN
select @count = count(*) from author
print @count
END
该触发器作用就是统计insert后的表数量,测试结果如下:
我们插入前数据条数
执行insert语句后,触发触发器,输出的条数是
(3)修改author的触发器为INSTEAD OF
alter TRIGGER authorTrigger ON author INSTEAD OF INSERT AS
declare @count int
BEGIN
select @count = count(*) from author
print @count
END
该触发器作用就是统计insert后的表数量,测试结果如下:
我们插入前数据条数
执行insert语句后,触发触发器,输出的条数是
看到这里,读者可能怀疑数据错了,第二种触发器INSTEAD OF为何插入后还是七条呢,不应该是八条吗
根据这个实验引出结论,
- after 触发器(insert、update、delete触发器)内的语句是在操作执行之后(已经作用在表上)才触发执行的
- instead of 触发器并不会执行操作(不会影响实际的表),它更像一个指令,遇到条件中的指令就触发了,就会执行触发器内的语句。
- 在执行 INSERT、UPDATE、MERGE 或 DELETE 语句的操作之后执行 AFTER 触发器
- INSTEAD OF 触发器可用于对一个或多个列执行错误或值检查,然后在插入、更新或删除行之前执行其他操作
无论after 触发器还是instead of 触发器,他们的作用都是在更新或更新后对表中数据操作,那么更新的旧数据该保存到哪里呢,sqlserver提供了两张临时表inserted 和 deleted 表
三、inserted 和 deleted 表
inserted表和deleted表对照
修改操作记录 inserted表 deleted表 增加(insert)记录 存放新增的记录 ............ 删除(deleted)记录 .............. 存放被删除的记录 修改(update)记录 存放更新后的记录 存放更新前的记录
- inserted表保存更新后的记录副本,deleted表保存更新前的记录副本。
- INSTEAD OF触发器和AFTER 触发器都可以使用inserted表和deleted表
- SQL Server 会自动创建和管理这两种表,用户使用它们作为条件,但是不能修改表中的数据
四、项目实战
开发中,有这么一个需求:当表中某些字段发生修改或者新增一条记录时,会自动更新表中modifyTime字段为当前时间,如何采用触发器实现呢?
分析:这里要注意是某些字段有更新,才更新modifyTime字段;如果你更新的字段不在指定的字段里,是不会更新modifyTime字段,答案如下:
CREATE TRIGGER modifMeterTrigger ON
Meter after UPDATE,INSERT AS
declare @upflag int
BEGIN
select
@upflag = case
when d.IsAddSecurityPlan != i.IsAddSecurityPlan then 1
when d.MeterVersion != i.MeterVersion then 1
when d.SystemNo != i.SystemNo then 1
when d.IsCancel != i.IsCancel then 1
when d.UseDate != i.UseDate then 1
when d.HankDate != i.HankDate then 1
when d.UseYear != i.UseYear then 1
when d.AddressCode != i.AddressCode then 1
when d.MeterType != i.MeterType then 1
when d.UserGasType != i.UserGasType then 1
else 0
end
from inserted i left join deleted d on d.id = i.id
if(@upflag > 0)
UPDATE Meter SET ModifTime = GETDATE( )
FROM Meter t INNER JOIN Inserted i ON t.id=i.id
END
看这个触发器定义语句,比之前的要复杂好多,我慢慢解读
(1)首先定义的After类型的触发器
(2)定义了一个局部变量,@upflag来标志更新的字段是否在指定的字段内
(3)将inserted和deleted采用左连接,通过case when来判断,如果inserted字段的值不等于deleted中的值,说明指定字段有更新,然后更新标志为@upflag=1
(4)最后判断@upflag是否为1,为1执行更新
(5)采用原表和Inserted内连接主要为了既可以批量更新也可以单条更新
注意一点:我们在更新可能会影响多行数据,如果我们在更新时采用id作为条件
比如如下触发器定义(来源于官方):
CREATE TRIGGER NewPODetail
ON Purchasing.PurchaseOrderDetail
AFTER INSERT AS
UPDATE PurchaseOrderHeader
SET SubTotal = SubTotal + LineTotal
FROM inserted
WHERE PurchaseOrderHeader.PurchaseOrderID = inserted.PurchaseOrderID ;
对于多行插入,示例的 DML 触发器可能不会正确运行,SQL官方提供了@@ROWCOUNT
函数来区分单行插入和多行插入
CREATE TRIGGER NewPODetail3
ON Purchasing.PurchaseOrderDetail
FOR INSERT AS
IF @@ROWCOUNT = 1
BEGIN
UPDATE Purchasing.PurchaseOrderHeader
SET SubTotal = SubTotal + LineTotal
FROM inserted
WHERE PurchaseOrderHeader.PurchaseOrderID = inserted.PurchaseOrderID
END
ELSE
BEGIN
UPDATE Purchasing.PurchaseOrderHeader
SET SubTotal = SubTotal +
(SELECT SUM(LineTotal)
FROM inserted
WHERE PurchaseOrderHeader.PurchaseOrderID
= inserted.PurchaseOrderID)
WHERE PurchaseOrderHeader.PurchaseOrderID IN
(SELECT PurchaseOrderID FROM inserted)
END;