公司之前一直存在一个规范,就是禁止嵌套事务的使用,一直不太明白为什么,试了下应该是无法控制回滚,今天看大牛的博客发现,问题远远不只如此。

具体总结下来是以下3个问题

1、内层事务回滚,只能回滚全部事务,无法控制单一事务回滚

2、内层事务提交后,回滚外层事务,也会把内层提交了的事务一起回滚

3、因为2的原因,只要整个事务不完全提交,日志空间都无法被释放


    嵌套事务可不会像其语法表现的那样看起来允许事务嵌套。我真不知道为什么有人会这样写代码,我唯一能够想到的就是某个哥们对SQL Server社区嗤之以鼻然后写了这样的代码说:“玩玩你们”。

    让我更详细的解释一下,SQL Server允许你在一个事务中开启嵌套另一个事务,SQL Server允许你提交这个嵌套事务,也允许你回滚这个事务。

    但是,嵌套事务并不是真正的“嵌套”,对于嵌套事务来说SQL Server仅仅能够识别外层的事务。嵌套事务是日志不正常增长的罪魁祸首之一因为开发人员以为回滚了内层事务,仅仅是回滚内层事务。

    但实际上当回滚内层事务时,会回滚整个内层事务,而不是仅仅是内层。这也是为什么我说嵌套事务并不存在。

    所以作为开发人员来讲,永远不要对事务进行嵌套。事务嵌套是邪恶的。

    如果你不相信我说的,那么通过下面的例子就就会相信。创建完数据库和表之后,每一条记录都会导致日志增加8K。

CREATE DATABASE NestedXactsAreNotReal; 
 GO 
 USE NestedXactsAreNotReal; 
 GO 
 ALTER DATABASE NestedXactsAreNotReal SET RECOVERY SIMPLE; 
 GO 
 CREATE TABLE t1 (c1 INT IDENTITY, c2 CHAR (8000) DEFAULT 'a'); 
 CREATE CLUSTERED INDEX t1c1 ON t1 (c1); 
 GO 
 SET NOCOUNT ON; 
 GO 
  测试 #1:回滚内部事务时仅仅回滚内部事务?
BEGIN TRAN OuterTran; 
 GO INSERT INTO t1 DEFAULT Values; 
 GO 1000 BEGIN TRAN InnerTran; 
 GO INSERT INTO t1 DEFAULT Values; 
 GO 1000 SELECT @@TRANCOUNT, COUNT (*) FROM t1; 
 GO

 

    你可以看到得出的结果是2和2000,下面我来回滚内部的事务,按照我们的猜想应该只回滚1000条吧,但事实上你会得到如下结果:

ROLLBACK TRAN InnerTran;
GO

消息 6401,级别 16,状态 1,第 2 行
无法回滚 InnerTran。找不到该名称的事务或保存点。

 

    好吧,由Books Online来看,我只能使用外部事务的名称或是将事务名称留空来进行回滚,代码如下:

ROLLBACK TRAN;
GO

SELECT @@TRANCOUNT, COUNT (*) FROM t1;
GO

 

    现在我得到结果是0和0。正如Books Online所言,这个回滚操作将外部事务进行了回滚并将全局变量@@TRANCOUNT设置为0。事务中所有的修改都被回滚,如果想部分回滚的话只能使用SAVE TRAN 和ROLLBACK TRAN。

   

测试 #2:嵌套事务中内部事务提交后会保存内部事务的修改吗?

BEGIN TRAN OuterTran; 
 GO BEGIN TRAN InnerTran; 
 GO INSERT INTO t1 DEFAULT Values; 
 GO 1000 COMMIT TRAN InnerTran; 
 GO SELECT COUNT (*) FROM t1; 
 GO

 

    正如我所期待,得到的结果是1000。这说明内部事务提交是会修改到磁盘的。但是如果这时外部事务回滚的话,那么不应该回滚内部事务…

ROLLBACK TRAN OuterTran; 
 GO SELECT COUNT (*) FROM t1; 
 GO

 

    但运行上面查询后结果是0,这说明外部事务的回滚会影响内部事务。

 

测试 #3:提交嵌套的事务的内部事务至少可以让我清除日志吧。

   在开始这个测试之前我首先清除了日志,然后运行如下代码:

BEGIN TRAN OuterTran; 
 GOBEGIN TRAN InnerTran; 
 GOINSERT INTO t1 DEFAULT Values; 
 GO 1000DBCC SQLPERF ('LOGSPACE'); 
 GO

   得到结果:

   

java事务嵌套 外层提交 内层回滚 事务嵌套失败会回滚吗_嵌套事务

 

    下面我将事务提交后运行CheckPoint(对于简单恢复模式的数据库将会截断日志),得到的结果:

COMMIT TRAN InnerTran; 
 GOCHECKPOINT; 
 GODBCC SQLPERF ('LOGSPACE'); 
 GO

   

java事务嵌套 外层提交 内层回滚 事务嵌套失败会回滚吗_java事务嵌套 外层提交 内层回滚_02

 

    我们发现日志的使用不减反赠,这是由于日志写入了CheckPoint记录(详情请看:How do checkpoints work and what gets logged)。提交内部事务不会导致日志被清除,这是由于外部事务回滚时也会连同内部事务一起回滚(译者注:所以这部分VLF在外部事务提交之前永远不会被标记位reusable)。所以这部分日志在外部事务提交之前永远不会被截断。为了证明这一点,我提交外部事务,然后再来看日志:

COMMIT TRAN OuterTran; 
 GOCHECKPOINT; 
 GODBCC SQLPERF ('LOGSPACE'); 
 GO

 

java事务嵌套 外层提交 内层回滚 事务嵌套失败会回滚吗_回滚_03