@@TranCount 之事务管理



在处理事务的时候,一般都用RollBack Transaction来回滚,但是如果在嵌套事务中这样使用的话,就会出现错误。
在SqlServer里,嵌套事务的层次是由@@TranCount全局变量反映出来的。每一次Begin Transaction都会引起@@TranCount加1。而每一次Commit Transaction都会使@@TranCount减1,而RollBack Transaction会回滚所有的嵌套事务包括已经提交的事务和未提交的事务,而使@@TranCount置0。例如:
Begin Transaction -- @@TranCount = 1
         BeginTransaction -- @@TranCount = 2  
                  BeginTransaction -- @@TranCount = 3
                  Commit Transaction -- @@TranCount = 2
         Commit Transaction -- @@TranCount = 1
Commit Transaction -- @@TranCount = 0
如果出现错误ROLLBACK TRANSACTION
则:
Begin Transaction -- @@TranCount = 1
         BeginTransaction -- @@TranCount = 2  
                  BeginTransaction -- @@TranCount = 3
                  ROLLBACK TRANSACTION  -- @@TranCount = 0
         Commit Transaction -- @@TranCount = 0---出现错误
Transaction count after EXECUTE indicates that a COMMIT or ROLLBACK TRANSACTION statement is missing. Previous count = 1, current count = 0.
         如果被嵌套的事务中发生错误,最简单的方法应该是无论如何都先将它提交,同时返回错误码(一个正常情况不可能出现的代码 如 -1)让上一层事务来处理这个错误,从而使@@TranCount 减1。 这样外层事务在回滚或者提交的时候能够保证外层事务在开始的时候和结束的时候保持一致。由于里层事务返回了错误码,因此外层事务(最外层)可以回滚事务,这样里面已经提交的事务也可以被回滚而不会出现错误。
         在项目中应该会常常出现这样的情况,一个存储过程里面用了事务,但是不能保证它会被别的带有事务的存储过程调用,如果单独调用的话,出现错误可以直接回滚,但是如果是被别的带事务的存储过程调用的话,RollBack 就会出错了。因此需要一种机制来区分,建立一个临时的变量来区分是否嵌套,和嵌套的层数,如下:
 

IF @@ERROR<>0
goto Error
 Commit Transaction<pre class="sql" name="code">DECLARE @TranCounter INT;
    SET @TranCounter = @@TRANCOUNT;
    IF @TranCounter > 0
        SAVE TRANSACTION ProcedureSave;
    ELSE搜索
        BEGIN TRANSACTION;
…………
--事务内要执行的代码
…………

Commit Transaction--下面返回要返回的值0只是个例子Return 0Error: IF @TranCounter = 0 ROLLBACK TRANSACTION; Else ROLLBACK TRANSACTION ProcedureSave; Return @Error

 

------------------------------------------------------------------

SELECT   '事务处理前',   @@TRANCOUNT             --值为   0   
  BEGIN   TRAN   
      SELECT   '第一个事务',   @@TRANCOUNT         --值为   1   
--          SELECT  'aa'  
          BEGIN   TRAN     
                SELECT   '第二个事务',   @@TRANCOUNT     --值为   2   
--                    SELECT   'bb'    
          COMMIT   TRAN   
          SELECT   '递交第二个事务',   @@TRANCOUNT   --值为   1   
  ROLLBACK   TRAN   
  SELECT   '回滚第一个事务',   @@TRANCOUNT   --值为   0   
    

 SELECT   '事务处理前',   @@TRANCOUNT             --值为   0   
  BEGIN   TRAN     
      SELECT   '第一个事务',   @@TRANCOUNT         --值为   1   
--          SELECT   'aa' 
      SAVE   TRAN   t1   
      SELECT   '保存第一个事务后',   @@TRANCOUNT   --值为   1     
          BEGIN   TRAN     
                SELECT   '第二个事务',   @@TRANCOUNT     --值为   2   
--                    SELECT   'bb'  
        ROLLBACK   TRAN   t1   
          SELECT   '回滚到保存点t1',   @@TRANCOUNT   --注意这里的值为   2     
  IF   @@TRANCOUNT>0   
  ROLLBACK   TRAN   
  SELECT   '处理结束',   @@TRANCOUNT   --为   0

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值