@@TranCount 之事务管理
来源:互联网 发布:大华淘宝客助手 编辑:程序博客网 时间:2024/06/05 10:44
在处理事务的时候,一般都用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<>0goto 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
- @@TranCount 之事务管理
- 事务管理之Spring事务管理
- 事务管理之Hibernate事务管理
- 事务管理之XA分布式事务管理
- @@TRANCOUNT (Transact-SQL)
- Spring声明式事务管理之xml事务管理
- Spring总结之事务管理
- Spring总结之事务管理
- spring之事务管理
- Spring之事务管理详解
- hibernate 之session 事务管理
- Mybatis深入之事务管理
- Spring之五 事务管理
- Spring注解之事务管理
- Spring事务管理之HibernateTransactionManager
- spring之事务管理1
- spring之事务管理2
- spring之事务管理
- 关于SpringMVC4使用过程中的路径问题,我们经常遇到的Controller未执行以及执行了之后出现404
- test10
- 将Vim改造为强大的IDE—Vim集成Ctags/Taglist/Cscope/Winmanager/NERDTree/OmniCppComplete(有图有真相)
- Windows7上搭建node和npm环境
- 0092 Java线程:Condition-线程通信更高效的方式【进阶】
- @@TranCount 之事务管理
- System UI 调试方法
- Cannot find executable for CFBundle
- Activity与Task
- 2014腾讯实习一面面试题
- SharpDevelop 使用笔记
- j_spring_security_check 404错误
- MySQL日期时间函数大全
- 类一定要定义拷贝构造函数,特别是在类成员含有指针的情况(不论指针是何种类型),安全!!!