0

这是一个奇怪的问题:有没有办法让回滚豁免插入到表中?

场景如下:我们有一个做事的触发器。

有时,这个触发器会调用RAISERRROR(). 并且外部事务回滚。

但是,在触发器中,我想将值插入到日志记录表中,并且在回滚期间不让它消失,如果你愿意的话,这是一个事务豁免插入。

4

3 回答 3

1

香蕉渔获

您可以捕获异常try-catch并将它们传递/抛出到外部范围,其中必须存在相同的东西。顺便说一句,这可以让您收集call-stack. 如果您仅通过存储过程工作,则可以实施此解决方案。每个 proc 都必须有这样的模式:

begin try
end try
begin catch
   if @@trancount > 0
     rollback

   insert into <log> (...)
   values (...)

   throw
end catch

所以最上面的过程将成功地在日志表中插入一行。

缺点:

  • 具有较深的层次结构 - 太多“虚拟”插入要回滚
  • 如果出现以下情况,仍然可以不记录任何内容:
    • 严重性太高
    • SP 不符合此模式
    • 有一个外部/“客户端”事务管理

优点:

  • 不完美,但仍然可行,实施起来也不难。

回馈:

  • 我确实在小型项目的生产服务器上使用了这种方法,确实收集了调用堆栈等等。这是一个后台软件,有大约 50-100 个在线用户,一些机器人在运行,远不是“高负载”。它运行良好,帮助解决了许多问题。开销没有打扰任何人。

难以捉摸的强盗

您可以构建一个插入日志表的 CLR 程序集,仅此而已……但是!可以指定一个单独的连接,使这个程序集通过这个单独的连接与 db 一起工作。这意味着 - 在单独的范围内。所以这个程序集的方法是从一个事务中被调用的,但是不管这个事务范围是被执行的。

所以,而不是:

using(SqlConnection connection = new SqlConnection("context connection=true"))

只需指定一个常规连接字符串。调用它后 - 抛出异常并修改ERROR_STATUS以避免额外记录相同的错误。

begin try
end try
begin catch
   if @@trancount > 0
     rollback

   if @@ERRROR_STATUS != @done_with_logging
     exec asm.log(...)

   raiserror @err_msg, @severity, @done_with_logging
end catch

缺点:

  • 可能会导致连接池问题
  • 获得微不足道的插入的有点长的方法
  • 可能会导致与建立连接和程序集本身相关的其他问题(+权限、所有权等)

优点:

  • 这个想法有点令人兴奋
  • 可以记录到文件而不是数据库

回馈:

  • 我不记得我是否在 prod 上使用过这种方法(尽管我记得这是最近的经历......或者只是为了测试目的而尝试实现它)

敲门,敲门,管理员

这个简单的语句(实际上只是附加选项WITH LOG)会将您想要的任何错误消息写入 SQL SERVER 事件日志:

RAISERROR(...) WITH LOG

这不是应该使用 SQL SERVER 日志的方式,但这是记录重要内容(用于解决问题)的最快方式。记录的事件可以在 SSMS 代理的窗口中查看。

缺点:

  • 可能有权限考虑
  • 糟糕的搜索和零定制能力
  • dba 会恨你的(当然,如果他曾经检查过服务器日志)

优点:

  • 单行实现

回馈:

  • 我只在一家公司遇到过这种登录方式。我想,前段时间它被几个开发人员用来捕捉细微的错误,但后来传播到代码中并成为“标准”。因此,一段时间后,几乎不可能在事件日志中找到特定的内容......实际上只有少数同事可以访问 prod 服务器的日志。所以它实际上是a)有害的b)无用的。因此,我一有时间就将其从系统的托管部分中删除。

飞翔的荷兰人

{基于 DML 触发器构建系统的基于意见的讨论的地方}

在我看来,您没有在项目中使用存储过程,而是执行临时查询。如果你有一个带有 ORM 或类似的后端应用程序 - 用它写日志。此外,也许这个后端应用程序是一个更好的地方来做你在那个触发器中做的事情。

如果您的项目是没有应用程序服务器/后端应用程序的客户端-服务器应用程序,而您得到的只是一个临时查询和触发器,那么需要记录的数据不多。没有调用堆栈(在服务器端)。而且很难确定您(用户、应用程序)是如何遇到这个特殊异常的。因此,在这种情况下,登录客户端可能更有用。

于 2018-09-25T22:57:05.633 回答
0

如果您创建一个表变量然后插入其中 - 这将不会包含在回滚中,因此您可以将内容转储到永久表中。

例如

declare @tab table (msg varchar(255))

BEGIN TRY

    BEGIN TRANSACTION

    select 1+2

    INSERT @tab values ('first step complete')

    SElect 1/0

    INSERT @tab values ('2nd step complete')

    COMMIT

END TRY

BEGIN CATCH

    ROLLBACK

    SELECT * FROM @tab
END CATCH
于 2018-09-26T15:11:07.647 回答
0

过段时间再写。您可以使用代替触发器:

create table t
(
i int,
s varchar(10)
)
go

create table t2
(
i int,
s varchar(10)
)
go

create table tlog
(
i int,
s varchar(10)
)
go

alter trigger tt on t
INSTEAD OF INSERT
AS
BEGIN
    rollback transaction
    raiserror ('something went wrong', 16, 2)
    insert tlog (i,s)
    select i, s
    from inserted
END
go

truncate table t
truncate table t2
truncate table tlog
go

select * from t
select * from t2
select * from tlog
go

begin transaction
insert t2 (i,s) values (1, 'abc')
insert t (i,s) values (1, 'abc')
commit transaction
go

select * from t
select * from t2
select * from tlog

给出以下输出:


(0 行受影响)


(0 行受影响)


(0 行受影响)

(1 行受影响)消息 50000,级别 16,状态 2,程序 tt,第 7 行出现问题

(1 行受影响) Msg 3609, Level 16, State 1, Line 4 事务在触发器中结束。该批次已中止。是


(0 行受影响)


(0 行受影响)


1个ABC

(1 行受影响)

于 2021-06-16T08:56:27.153 回答