SQL Server嵌套事务
一、@@TRANCOUNT
在将事务前,我们先来了解一下@@TRANCOUNT ,@@trancount返回上传执行begin transaction语句的事务计数。
1、每执行一次begin transaction语句@@trancount 将增加1。
2、执行rollback transaction 语句@@trancount将变为0,但执行rollback transaction savepoint_name语句@@trancount不会有影响。
declare @currntTranCount VARCHAR
set @currntTranCount=@@TRANCOUNT
print '未执行事务前全局@@TRANCOUNT:'+@currntTranCount begin transaction
set @currntTranCount=@@TRANCOUNT
print '执行事务后全局@@TRANCOUNT:'+@currntTranCount rollback transaction
set @currntTranCount=@@TRANCOUNT
print '回滚事务后全局@@TRANCOUNT:'+@currntTranCount
输出:
未执行事务前全局@@TRANCOUNT:0
执行事务后全局@@TRANCOUNT:1
回滚事务后全局@@TRANCOUNT:0
declare @currntTranCount VARCHAR
set @currntTranCount=@@TRANCOUNT
print '未执行事务前全局@@TRANCOUNT:'+@currntTranCount begin transaction
set @currntTranCount=@@TRANCOUNT
print '执行事务后全局@@TRANCOUNT:'+@currntTranCount save transaction savePoint_Tran1
set @currntTranCount=@@TRANCOUNT
print '保存事务点后全局@@TRANCOUNT:'+@currntTranCount rollback transaction savePoint_Tran1 --@@TRANCOUNT没受影响
set @currntTranCount=@@TRANCOUNT
print '回滚事务保存点后全局@@TRANCOUNT:'+@currntTranCount rollback transaction
set @currntTranCount=@@TRANCOUNT
print '回滚事务后全局@@TRANCOUNT:'+@currntTranCount
输出:
未执行事务前全局@@TRANCOUNT:0
执行事务后全局@@TRANCOUNT:1
保存事务点后全局@@TRANCOUNT:1
回滚事务保存点后全局@@TRANCOUNT:1
回滚事务后全局@@TRANCOUNT:0
3、执行commit transaction或commit work语句@@trancount将递减1。
declare @currntTranCount VARCHAR
set @currntTranCount=@@TRANCOUNT
print '未执行事务前全局@@TRANCOUNT:'+@currntTranCount begin transaction
set @currntTranCount=@@TRANCOUNT
print '执行事务后全局@@TRANCOUNT:'+@currntTranCount commit transaction
set @currntTranCount=@@TRANCOUNT
print '提交事务后全局@@TRANCOUNT:'+@currntTranCount
输出:
未执行事务前全局@@TRANCOUNT:0
执行事务后全局@@TRANCOUNT:1
提交事务后全局@@TRANCOUNT:0
4、ROLLBACK TRANSACTION (Transact-SQL):https://msdn.microsoft.com/zh-cn/library/ms181299.aspx
5、COMMIT TRANSACTION (Transact-SQL):https://msdn.microsoft.com/zh-cn/library/ms190295.aspx
二、嵌套事务
1、回滚嵌套事务:rollback transaction只能回滚最外面的事务名称或rollback transaction不指定某个事务名进行回滚;
如果执行rollback transaction transaction_innerTran,SQL会提示“无法回滚 transaction_innerTran。找不到该名称的事务或保存点。”,原因是transaction_innerTran不是最外层的事务;也就是说:回滚事务一回滚就所有事务都被回滚。
rollback transaction可以回滚某个事务保存点(SAVE TRANSACTION savePoint_Tran), 如ROLLBACK TRAN savePoint_Tran,但是回滚事务保存点不会使事务计数@@TRANCOUNT减少。
1)内层回滚事务存储过程
--嵌套事务:内层事务
CREATE PROCEDURE pro_InnerTransactionTest
AS
BEGIN
declare @currntTranCount VARCHAR(50)
set @currntTranCount=@@TRANCOUNT
print '未执行内层事务前全局@@TRANCOUNT:'+@currntTranCount begin transaction transaction_innerTran
set @currntTranCount=@@TRANCOUNT
print '执行内层事务后全局@@TRANCOUNT:'+@currntTranCount rollback transaction transaction_innerTran
--rollback transaction
--rollback transaction transaction_outerTran --rollback都报错
set @currntTranCount=@@TRANCOUNT
print '回滚内层事务后全局@@TRANCOUNT:'+@currntTranCount return 0;
--return 1;
END
GO
2)外层开启事务执行
--嵌套事务:外层执行
declare @currntTranCount varchar(50);
declare @result int;
set @currntTranCount=@@TRANCOUNT;
print '未执行外层事务前全局@@TRANCOUNT:'+@currntTranCount; begin transaction transaction_outerTran;
set @currntTranCount=@@TRANCOUNT;
print '执行外层事务后全局@@TRANCOUNT:'+@currntTranCount; execute @result = pro_InnerTransactionTest; if(@result <= 0)
begin
rollback transaction transaction_outerTran;
set @currntTranCount=@@TRANCOUNT;
print '回滚外层事务后全局@@TRANCOUNT:'+@currntTranCount; return;
end
commit transaction transaction_outerTran
set @currntTranCount=@@TRANCOUNT;
print '提交外层事务后全局@@TRANCOUNT:'+@currntTranCount;
输出:
未执行外层事务前全局@@TRANCOUNT:0
执行外层事务后全局@@TRANCOUNT:1
未执行内层事务前全局@@TRANCOUNT:1
执行内层事务后全局@@TRANCOUNT:2
消息 6401,级别 16,状态 1,过程 pro_InnerTransactionTest,第 13 行
无法回滚 transaction_innerTran。找不到该名称的事务或保存点。
回滚内层事务后全局@@TRANCOUNT:2
消息 266,级别 16,状态 2,过程 pro_InnerTransactionTest,第 0 行
EXECUTE 后的事务计数指示 BEGIN 和 COMMIT 语句的数目不匹配。上一计数 = 1,当前计数 = 2。
回滚外层事务后全局@@TRANCOUNT:0
2、提交嵌套事务:commit transaction 可以单独指定某个事务名,如transaction_outerTran,transaction_innerTran进行提交,但即使transaction_innerTran提交成功了,只要最外面的事务transaction_outerTran回滚,transaction_innerTran提交的数据也会被回滚的。
1)内层提交事务存储过程
--嵌套事务:内层事务
CREATE PROCEDURE pro_InnerTransactionTest
AS
BEGIN
declare @currntTranCount VARCHAR(50)
set @currntTranCount=@@TRANCOUNT
print '未执行内层事务前全局@@TRANCOUNT:'+@currntTranCount begin transaction transaction_innerTran
set @currntTranCount=@@TRANCOUNT
print '执行内层事务后全局@@TRANCOUNT:'+@currntTranCount --rollback transaction transaction_innerTran
--rollback transaction
--rollback transaction transaction_outerTran --rollback都报错
--set @currntTranCount=@@TRANCOUNT
--print '回滚内层事务后全局@@TRANCOUNT:'+@currntTranCount commit transaction transaction_innerTran --提交内层事务,外层回滚或提交事务都没报错 return 0;
--return 1;
END
GO
2)外层开启事务执行后输出:
未执行外层事务前全局@@TRANCOUNT:0
执行外层事务后全局@@TRANCOUNT:1
未执行内层事务前全局@@TRANCOUNT:1
执行内层事务后全局@@TRANCOUNT:2
回滚外层事务后全局@@TRANCOUNT:0 (或:提交外层事务后全局@@TRANCOUNT:0)
三、解决办法:
1)内层存储过程
--嵌套事务:内层事务
CREATE PROCEDURE pro_InnerTransactionTest
AS
BEGIN
declare @currntTranCount varchar(50)
declare @sumError int=0
declare @isSingleTran bit=1 --是否单个事务而非嵌套事务 set @currntTranCount=@@TRANCOUNT
print '未执行内层事务前全局@@TRANCOUNT:'+@currntTranCount SET XACT_ABORT ON --设置事务回滚到原点
--开始事务
if (@currntTranCount=0)
begin
begin transaction transaction_innerTran
set @isSingleTran=1;
set @currntTranCount=@@TRANCOUNT
print '执行内层事务后全局@@TRANCOUNT:'+@currntTranCount
end
else
begin
save transaction savepoint_innerTran --保存事务点
set @isSingleTran=0;
set @currntTranCount=@@TRANCOUNT
print '保存内层事务点全局@@TRANCOUNT:'+@currntTranCount
end --UPDATE [dbo].[FinanceInfo] SET [Balance] = [Balance]+1000 WHERE [UserId] = 10001
--set @sumError = @sumError + @@error
--UPDATE [dbo].[FinanceInfo] SET [Balance] = [Balance]-1000 WHERE [UserId] = 10002
--set @sumError = @sumError + @@error set @sumError=1; --手动测试 if(@sumError = 0)
begin
if(@isSingleTran = 1)
begin
commit transaction transaction_innerTran
set @currntTranCount=@@TRANCOUNT
print '提交内层事务点全局@@TRANCOUNT:'+@currntTranCount
end
else
begin
set @currntTranCount=@@TRANCOUNT
print '返回内层全局@@TRANCOUNT:'+@currntTranCount
return 1; --成功
end
end
else
begin
if(@isSingleTran = 1)
begin
rollback transaction --发生错误,回滚事务
set @currntTranCount=@@TRANCOUNT
print '回滚内层事务全局@@TRANCOUNT:'+@currntTranCount
end
else
begin
rollback transaction savepoint_innerTran
set @currntTranCount=@@TRANCOUNT
print '回滚内层事务点全局@@TRANCOUNT:'+@currntTranCount
return 0; --失败
end
end
END
GO
2)外层开启事务执行存储过程
--嵌套事务:外层执行
declare @currntTranCount varchar(50);
declare @result int;
set @currntTranCount=@@TRANCOUNT;
print '未执行外层事务前全局@@TRANCOUNT:'+@currntTranCount; begin transaction transaction_outerTran;
set @currntTranCount=@@TRANCOUNT;
print '执行外层事务后全局@@TRANCOUNT:'+@currntTranCount; execute @result = pro_InnerTransactionTest; if(@result <= 0)
begin
rollback transaction transaction_outerTran;
set @currntTranCount=@@TRANCOUNT;
print '回滚外层事务后全局@@TRANCOUNT:'+@currntTranCount; return;
end
commit transaction transaction_outerTran
set @currntTranCount=@@TRANCOUNT;
print '提交外层事务后全局@@TRANCOUNT:'+@currntTranCount; --单个事务执行
declare @result INT
execute @result = pro_InnerTransactionTest;
select @result
SQL Server嵌套事务的更多相关文章
- sql server 学习笔记 (nested transaction 嵌套事务)
什么时候会用到嵌套事务 ? 为了代码复用,我们会写许多的储蓄过程,而中间如果需要使用到 transaction 难免就会发生嵌套了. sql server 并不直接支持嵌套事务. 但它可以用一些招式来 ...
- sql server中嵌套事务*
转自 https://www.cnblogs.com/guanjie20/archive/2013/02/17/2914488.html 我们在写事务时经常遇到的问题如下: 消息 266,级别 16, ...
- SQL Server 2014 新特性——内存数据库
SQL Server 2014 新特性——内存数据库 目录 SQL Server 2014 新特性——内存数据库 简介: 设计目的和原因: 专业名词 In-Memory OLTP不同之处 内存优化表 ...
- SQL Server简洁查询正在运行的进程SQL
通常我们可以使用 sp_who2 我们希望更加简洁的信息,下面这个查询使用系统表sys.sysprocesses,以及sys.dm_exec_sql_text做OUTER APPLY. T-SQL是这 ...
- SQL SERVER 分布式事务(DTC)
BEGIN DISTRIBUTED TRANSACTION指定一个由 Microsoft 分布式事务处理协调器 (MS DTC) 管理的 Transact-SQL 分布式事务的起始. 语法BEGIN ...
- SQL Server 全局变量
SQL Server中所有全局变量都使用两个@符号作为前缀 --1.@@error 最后一个T-SQL错误的错误号(目的是或得违反约束的错误号) insert into Subject values( ...
- sql server 分布式事务
使用分布式事务刚好可以解决集群同时更新多台SQL SERVER数据库,要么全部成功,要么全部回滚的需要. 原来微软早考虑到此方面的问题了. 下面背书,贴出微软官网上面的帮助文档: 分布式事务跨越两个或 ...
- (转)SQLServer_十步优化SQL Server中的数据访问 二
原文地址:http://tech.it168.com/a2009/1125/814/000000814758_all.shtml 第五步:识别低效TSQL,采用最佳实践重构和应用TSQL 由于每个程序 ...
- SQL Server 数据库try catch 存储过程
SQL Server 在生产环境中这样写存储过程的坑都避免了吗? 原文链接: http://www.cnblogs.com/chenmh/p/7856777.html 概述 最近因为业务的需求写了一段 ...
随机推荐
- HTML入门8
今天开始接触HTML里面的多媒体和嵌入内容 前面只讲了文字,下面来讲能够让网页动起来,更加有趣的嵌入元素,包含多媒体,包含图像的不同方式,以及怎样嵌入视频. HTML中图片,下面将深入使用它,以及&l ...
- HTML5_canvas_图片加载_双缓冲_跳帧闪烁问题
canvas 图片加载 pen.drawImage(ele, showX, showY, imgWidth, imgHeight); ele 将 img 元素 加载到画布上 步骤 1. 创建一个 ...
- javaweb中的乱码问题(初次接触时写)
javaweb中的乱码问题 在初次接触javaweb中就遇到了乱码问题,下面是我遇到这些问题的解决办法 1. 页面乱码(jsp) 1. 在页面最前方加上 <%@ page language=&q ...
- if-else案例–开关灯
首先,创建一个html页面,添加一个div盒子,用css设置相应的样式,用js获取盒子的元素,通过点击事件,设置body的背景颜色,用if..else来判断当什么状态设置相应的颜色,(swith... ...
- XAMPP 安装时 MySQL 无法启动,且提示端口占用。
今天安装XAMPP时遇到了几个坑,忙活了一上午才搞定,写下来分享给同样遇坑的盆友们. MySQL 点击start 提示端口3306被占用,我改了端口号,又改了注册表,将注册表地址改为xampp中mys ...
- java课程设计团队博客《基于学院的搜索引擎》
JAVA课程设计 基于学院网站的搜索引擎 对学院网站用爬虫进行抓取.建索(需要中文分词).排序(可选).搜索.数据摘要高亮.分页显示.Web界面. 一.团队介绍 学号 班级 姓名 简介 2016211 ...
- 移动端无法复制:使用clipboard.js碰到的一个小问题
移动端无法复制:使用clipboard.js碰到的一个小问题 直接看下面的代码:在移动端访问,点击,能正常复制. <html> <head> <meta http-e ...
- oo第二单元作业总结
oo第二单元博客总结 在第一单元求导结束后,迎来了第二单元的多线程电梯的问题,在本单元前两次作业中个人主要应用两个线程,采用“生产者-消费者”模式和共享数据变量的方式解决问题.在第三次作业中加入多个电 ...
- [dev][ipsec] netlink是什么
介绍: https://www.linuxjournal.com/article/7356 大纲: man手册 http://man7.org/linux/man-pages/man7/netlink ...
- JAVA RPC(二)序列化协议杂谈
序列化和反序列化作为Java里一个较为基础的知识点,大家心里也有那么几句要说的,但我相信很多小伙伴掌握的也就是那么几句而已,如果再深究问一下Java如何实现序列化和反序列化的,就可能不知所措了!遥记当 ...