7

如何防止在同一个表上的同一事件中触发的两个触发器之间的锁定问题?

我正在处理的数据库已经有一个加密的更新触发器,因此我无法修改它。我做了另一个更新触发器来完成一些新任务,当我直接在数据库上测试它时它工作正常,但当我在前端应用程序上对产品进行更新时失败。显然,当我激活触发器时,两个触发器都失败了。我收到的消息类似于“文档已打开,我将增加它的价值”。

这是锁定问题吗?

有一个相关的问题,有人说我们可以在一张桌子上拥有多个触发器(针对同一事件)。

这是我的触发器代码:

SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO

CREATE TRIGGER [dbo].[tr_st_rep_update]
ON [dbo].[st]
AFTER UPDATE
AS
  BEGIN
      SET NOCOUNT ON;

      IF ( update(ref)
            OR update(design)
            OR update(u_update)
            OR update(u_ativo)
            OR update(stock)
            OR update(epv1)
            OR update(epv2)
            OR update(epv3)
            OR update(peso)
            OR update(u_catnv1)
            OR update(u_catnv2)
            OR update(u_catnv3)
            OR update(u_dpromoi)
            OR update(u_dpromof)
            OR update(u_destaque) )
        BEGIN
            IF (SELECT count(*)
                FROM   Inserted
                       INNER JOIN Deleted
                         ON Inserted.ststamp = Deleted.ststamp
                WHERE  inserted.u_ativo = 1
                        OR ( Deleted.u_ativo = 1
                             AND Inserted.u_ativo = 0 )) > 0
              BEGIN
                  INSERT INTO RepData
                              (id,
                               REF,
                               familia,
                               stock,
                               epv1,
                               epv2,
                               epv3,
                               peso,
                               u_accao,
                               imagem,
                               process)
                  SELECT Inserted.ststamp AS id,
                         Inserted.REF     AS REF,
                         Inserted.familia AS familia,
                         Inserted.stock   AS stock,
                         Inserted.epv1    AS epv1,
                         Inserted.epv2    AS epv2,
                         Inserted.epv3    AS epv3,
                         Inserted.peso    AS peso,
                         CASE
                           WHEN Deleted.u_ativo = 1
                                AND Inserted.u_ativo = 0 THEN 'd'
                           ELSE 'u'
                         END              AS u_accao,
                         Inserted.imagem  AS imagem,
                         0                AS process
                  FROM   Inserted
                         INNER JOIN Deleted
                           ON Deleted.ststamp = Inserted.ststamp
                  WHERE  inserted.u_ativo = 1
                          OR ( Deleted.u_ativo = 1
                               AND Inserted.u_ativo = 0 )
              END
        END
  END 

任何帮助,将不胜感激。

更新:数据库是 MSSQL 2008

4

2 回答 2

15

使用触发器进行部署后的现场开发和定制是一个诱人的,但一开始是个坏主意,并且无疑会不断地为您产生类似的问题。

但是,考虑到这一点,那么 First: 表可以有多个触发器,这不是问题所在。

其次,错误消息“文档已打开,我将增加它的值”来自您的客户端应用程序或来自其他(加密)触发器,它不是 SQL Server 错误消息。鉴于此,您可能可以尝试将加密触发器设置为首先执行,或者将您的触发器设置为最后执行。这可能无法解决问题,但它可能会将错误从加密的触发器转移到您的触发器中,您有更好的机会以可管理的方式报告和/或解决它。

虽然是临时的,但唯一可以从您的触发器中看到的问题是,如果另一个触发器也在写入RepData表,并且您的双重写入导致重复键违规。


触发顺序可以通过sp_settriggerorder系统程序来控制,这里记录。

于 2013-01-18T20:30:59.277 回答
1

问题解决了。

我真的不知道问题的根源,尽管我认为这与表锁定有关,在这种情况下是在 Inserted 表上。

我只是更改了内部选择语句,以便直接从 st 表中获取值,而不是从 Inserted 表中获取值。

谢谢大家。

于 2013-01-18T19:43:59.150 回答