存储过程:EXECUTE 后的事务计数指示 BEGIN 和 COMMIT 语句的数目不匹配。上一计数 = 1,当前计数 = 0

本文主要是介绍存储过程:EXECUTE 后的事务计数指示 BEGIN 和 COMMIT 语句的数目不匹配。上一计数 = 1,当前计数 = 0,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

今天遇到个情况,在数据库客户端上单独执行(或调试模式)存储过程,都不会抛出异常信息。

但是通过C#程序,调用数据接口,数据接口是使用 SqlSugar 在C#中创建了一个事务(嵌套1个存储过程),如果存储过程内部出错,触发了存储过程自身的ROLLBACK,程序端就会接受到抛出的异常信息:EXECUTE 后的事务计数指示 BEGIN 和 COMMIT 语句的数目不匹配。上一计数 = 1,当前计数 = 0

经过资料查询和研究,找到一位博友的阐述如下:

因为存储过程的ROLLBACK命令清空了@@TRANCOUNT,该全局变量存储了当前连接中的事务数量,每当执行BEGIN TRAN命令时,该变量加1,每当执行COMMIT TRAN命令时,该变量减1,每当执行ROLLBACK TRAN命令或者ROLLBACK TRAN POINT_NAME时(没有保存点时),该变量清空。如果执行ROLLBACK TRAN POINT_NAME且有保存点POINT_NAME时,@@TRANCOUNT不变,但保存点POINT_NAME失效,再次ROLLBACK TRAN POINT_NAME时依旧是清空@@TRANCOUNT。

上面那段很饶,但全是干货,可以细细咀嚼。

而c#事务嵌入存储过程事务进行执行之后,因为触发了存储过程内部的ROLLBACK TRAN,所以@@TRANCOUNT清空为0,导致c#代码创建的事务再进行下一步操作时找不到对应的事务了,所以就报出了上面的异常。(具体里面的机制,有兴趣的可以深入研究一下)

面对这种问题,有一种解决方案,那就是在创建存储过程时,尽量在执行存储过程的事务之前先判断一下自身是否被嵌入在另外一个事务之内。如果未被嵌入在另外一个事务之内,则使用BEGIN TRAN POINT_NAME来开启一个事务。如果被嵌入在另外一个事务之内,则使用SAVE TRAN POINT_NAME来创建一个保存点。无论是开启事务还是设置保存点,其回滚都可以写为:ROLLBACK TRAN POINT_NAME,执行该回滚命令的区别就是,当为保存点模式时,@@TRANCOUNT不变,不会对存储过程自身的上级事务产生影响。当为开启事务模式时,也就意味着自身上级并没有事务,那么清空@@TRANCOUNT自然也就无所谓了。

--结构示例
--开启事务之前先获取当前的事务数量@@TRANCOUNT
DECLARE @T_COUNT INTSET @T_COUNT = @@TRANCOUNTIF(@T_COUNT > 0)--有上层事务SAVE TRAN POINT_NAME--创建保存点
ELSE--无上层事务BEGIN TRAN POINT_NAME--开启新事务
--处理过程
--如果无异常IF(@T_COUNT > 0)--有上层事务--不作任何提交处理,提交处理交由上层事务
ELSE--无上层事务COMMIT TRAN POINT_NAME--直接提交--如果有异常
ROLLBACK TRAN POINT_NAME--执行回滚,如果有保存点,则意味着存在上层事务,则回滚到保存点,@@TRANCOUNT清空。如果无保存点,则说明无上层事务,则将事务全部回滚,并将@@TRANCOUNT清空为0

这篇关于存储过程:EXECUTE 后的事务计数指示 BEGIN 和 COMMIT 语句的数目不匹配。上一计数 = 1,当前计数 = 0的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/833560

相关文章

C++使用栈实现括号匹配的代码详解

《C++使用栈实现括号匹配的代码详解》在编程中,括号匹配是一个常见问题,尤其是在处理数学表达式、编译器解析等任务时,栈是一种非常适合处理此类问题的数据结构,能够精确地管理括号的匹配问题,本文将通过C+... 目录引言问题描述代码讲解代码解析栈的状态表示测试总结引言在编程中,括号匹配是一个常见问题,尤其是在

JavaScript中的reduce方法执行过程、使用场景及进阶用法

《JavaScript中的reduce方法执行过程、使用场景及进阶用法》:本文主要介绍JavaScript中的reduce方法执行过程、使用场景及进阶用法的相关资料,reduce是JavaScri... 目录1. 什么是reduce2. reduce语法2.1 语法2.2 参数说明3. reduce执行过程

redis群集简单部署过程

《redis群集简单部署过程》文章介绍了Redis,一个高性能的键值存储系统,其支持多种数据结构和命令,它还讨论了Redis的服务器端架构、数据存储和获取、协议和命令、高可用性方案、缓存机制以及监控和... 目录Redis介绍1. 基本概念2. 服务器端3. 存储和获取数据4. 协议和命令5. 高可用性6.

在MySQL执行UPDATE语句时遇到的错误1175的解决方案

《在MySQL执行UPDATE语句时遇到的错误1175的解决方案》MySQL安全更新模式(SafeUpdateMode)限制了UPDATE和DELETE操作,要求使用WHERE子句时必须基于主键或索引... mysql 中遇到的 Error Code: 1175 是由于启用了 安全更新模式(Safe Upd

PLsql Oracle 下载安装图文过程详解

《PLsqlOracle下载安装图文过程详解》PL/SQLDeveloper是一款用于开发Oracle数据库的集成开发环境,可以通过官网下载安装配置,并通过配置tnsnames.ora文件及环境变... 目录一、PL/SQL Developer 简介二、PL/SQL Developer 安装及配置详解1.下

在Java中使用ModelMapper简化Shapefile属性转JavaBean实战过程

《在Java中使用ModelMapper简化Shapefile属性转JavaBean实战过程》本文介绍了在Java中使用ModelMapper库简化Shapefile属性转JavaBean的过程,对比... 目录前言一、原始的处理办法1、使用Set方法来转换2、使用构造方法转换二、基于ModelMapper

springboot启动流程过程

《springboot启动流程过程》SpringBoot简化了Spring框架的使用,通过创建`SpringApplication`对象,判断应用类型并设置初始化器和监听器,在`run`方法中,读取配... 目录springboot启动流程springboot程序启动入口1.创建SpringApplicat

本地搭建DeepSeek-R1、WebUI的完整过程及访问

《本地搭建DeepSeek-R1、WebUI的完整过程及访问》:本文主要介绍本地搭建DeepSeek-R1、WebUI的完整过程及访问的相关资料,DeepSeek-R1是一个开源的人工智能平台,主... 目录背景       搭建准备基础概念搭建过程访问对话测试总结背景       最近几年,人工智能技术

Redis存储的列表分页和检索的实现方法

《Redis存储的列表分页和检索的实现方法》在Redis中,列表(List)是一种有序的数据结构,通常用于存储一系列元素,由于列表是有序的,可以通过索引来访问元素,因此可以很方便地实现分页和检索功能,... 目录一、Redis 列表的基本操作二、分页实现三、检索实现3.1 方法 1:客户端过滤3.2 方法

关于Gateway路由匹配规则解读

《关于Gateway路由匹配规则解读》本文详细介绍了SpringCloudGateway的路由匹配规则,包括基本概念、常用属性、实际应用以及注意事项,路由匹配规则决定了请求如何被转发到目标服务,是Ga... 目录Gateway路由匹配规则一、基本概念二、常用属性三、实际应用四、注意事项总结Gateway路由