1

我目前正在创建一个 SQL Server 触发器,它在更新后应该填充一个更改日志,其中包含所做的任何工资更改以及由哪个用户更改。

我制作的应该接收更新的变更日志表可以在这里看到:

CREATE TABLE ChangeLog
(
ChangeID int identity(1,1) primary key,
EmpID int,
[User] nvarchar(30),
[Date] smalldatetime,
OldRate money,
NewRate money
)

据我所知,表的一切都很好,但是当我去创建触发器时,我开始收到一些错误(以及一大堆混乱。)我已经尝试研究几种做触发器的方法用新行更新不同的表,据我所知,它的结构是正确的,但是我不确定我是如何弄乱内容的。

CREATE TRIGGER payrate_change ON Employees FOR UPDATE
As
DECLARE @originalPay VARCHAR(100)
DECLARE @newPay VARCHAR(100)
If UPDATE (PayRate)
BEGIN
SELECT @originalPay = (SELECT PayRate FROM deleted)
SELECT @newPay = (SELECT PayRate FROM inserted)
INSERT INTO [dbo].[ChangeLog]
        (ChangeID, EmpID, [User], [Date], OldRate, NewRate)
    VALUES
        identity(1,1), 
        Employees.EmployeeID, 
        SUSER_SNAME(), 
        SYSDATETIME(), 
        @originalPay, 
        @newPay
    FROM inserted
END

两个 select 语句的 payrate 部分给出了一个错误,指出它们是无效的列,即使它上面的两行调用 PayRate 就好了。然后对于 Insert Into 语句的值部分,尽管我在创建表时以完全相同的方式使用它,但 identity 声称语法不正确。

有人对此触发器可能存在什么问题有任何想法吗?任何投入将不胜感激。

编辑:

在下面建议的更改之后,我将 ChangeID(主键)重新添加到触发器中,但是现在 IDENTITY 给了我一个不正确的语法错误:

CREATE TRIGGER payrate_change ON Employees FOR UPDATE
As

If UPDATE (PayRate)
BEGIN

INSERT INTO [dbo].[ChangeLog]
        (ChangeID, EmpID, [User], [Date], OldRate, NewRate)
    select
        IDENTITY(1,1),
        i.EmployeeID, 
        SUSER_SNAME(), 
        SYSDATETIME(), 
        d.PayRate, 
        i.PayRate
    FROM inserted i
    JOIN deleted d on i.EmployeeID = d.deleted
END

EDIT2:为有兴趣的人添加最终结果:

CREATE TRIGGER payrate_change2 ON Employees FOR UPDATE
AS

If UPDATE (Pay)
BEGIN

INSERT INTO [dbo].[ChangeLog]
        (EmpID, [User], [Date], OldRate, NewRate)
    select
        i.EmployeeID, 
        SUSER_SNAME(), 
        SYSDATETIME(), 
        d.Pay, 
        i.Pay
    FROM inserted i
    JOIN deleted d on i.EmployeeID = d.EmployeeID
END
4

1 回答 1

0

省略标识列,它将自行填充

您的触发器不处理多行更新,请查看最佳实践:为多行操作编码 SQL Server 触发器

你可以JOIN删除和插入伪表,我假设EmployeeID是关键?

试试这个

CREATE TRIGGER payrate_change ON Employees FOR UPDATE
As

If UPDATE (PayRate)
BEGIN

INSERT INTO [dbo].[ChangeLog]
        (EmpID, [User], [Date], OldRate, NewRate)
    select
        i.EmployeeID, 
        SUSER_SNAME(), 
        SYSDATETIME(), 
        d.PayRate, 
        i.PayRate
    FROM inserted i
    JOIN deleted d on i.EmployeeID = d.deleted
END

另外请注意,If UPDATE (PayRate)即使您执行类似的操作也会触发SET PayRate = PayRate 因此您可能想检查该值是否实际已更改,您可以在插入和删除之间的 JOIN 中添加它,并确保检查 NULLS

于 2013-02-19T00:12:34.117 回答