0

0

如何在.NET中实现Oracle事务的提交与回滚_OracleTransaction对象与隔离级别配置

P粉602998670

P粉602998670

发布时间:2026-03-15 14:08:03

|

769人浏览过

|

来源于php中文网

原创

OracleTransaction.Commit()未生效的根本原因是连接被意外关闭或复用,导致Dispose()时强制回滚;务必确保Commit()在Connection仍打开且未释放前执行。

OracleTransaction.Commit() 为什么没生效?

常见现象是调用 commit() 后查数据库发现数据没变,或事务中途被自动提交。根本原因通常是连接被意外复用或提前关闭——oracleconnectiondispose() 或离开 using 块时,会强制回滚未完成的事务,哪怕你已经调用了 commit()

实操建议:

灵机语音
灵机语音

灵机语音

下载
  • 确保 OracleTransaction 和它所属的 OracleConnection 生命周期严格对齐:事务必须在连接仍打开且未释放前完成 Commit()Rollback()
  • 不要把 OracleConnection 存在静态字段或长生命周期对象里;每次事务都应新建或从连接池安全获取
  • 避免在 try 块里只捕获特定异常而漏掉 OracleException 或连接中断类错误,导致事务状态悬空
  • 示例中易错写法:
    using (var conn = new OracleConnection(connStr)) {
        conn.Open();
        using (var tx = conn.BeginTransaction()) {
            // ... 执行命令
            tx.Commit(); // ✅ 正确
        } // ❌ 这里 conn 还活着,tx 已 Dispose,但没问题
    }
    但如果把 conn 提前 Close()Dispose()Commit() 就会抛 InvalidOperationException

BeginTransaction(isolationLevel) 隔离级别不生效?

Oracle 对 SQL 标准隔离级别的支持有限:ReadCommitted 是默认且唯一完全支持的;Serializable 被映射为 Oracle 的 SELECT FOR UPDATE 行为,并非真正的可序列化;RepeatableReadReadUncommitted 会被静默降级为 ReadCommitted,不报错也不警告。

实操建议:

  • 别依赖 RepeatableRead 实现两次读一致——Oracle 底层靠多版本并发控制(MVCC),同一事务内普通 SELECT 天然可重复读,无需额外配置
  • 真需要串行化语义,用 SELECT ... FOR UPDATE 显式加锁,而不是靠事务级别
  • 检查实际生效级别:执行后查 V$TRANSACTION 视图的 ISOLATION_LEVEL 字段,或捕获 OracleCommand 执行时的警告信息
  • 代码中显式指定更安全:
    var tx = conn.BeginTransaction(IsolationLevel.ReadCommitted);
    比无参调用更明确,也方便后续审计

嵌套事务和 Savepoint 怎么用才可靠?

.NET 的 OracleTransaction 不支持真正嵌套事务(即 BeginTransaction()BeginTransaction()),所谓“嵌套”只是创建保存点(Savepoint)。一旦外层事务回滚,所有保存点失效;但回滚到某个保存点,不会影响外层事务状态。

实操建议:

  • tx.Save("sp1") 创建保存点,用 tx.Rollback("sp1") 回滚到它——注意名称区分大小写,且不能重名
  • 保存点不是独立事务,不能提交,只能回滚;也不能跨连接或跨命令生命周期存在
  • 高频陷阱:在循环中反复创建同名保存点,导致后一次覆盖前一次,Rollback("sp1") 只能回到最后一次设置的位置
  • 适合场景:单个业务操作中局部失败需部分撤回(如批量插入时某条违反约束),而非替代分布式事务

OracleException 中哪些错误必须触发 Rollback?

不是所有异常都要回滚。Oracle 报错分三类:连接类(如 ORA-12154)、语法类(ORA-00900)、事务类(ORA-02091、ORA-08176)。只有影响事务一致性的错误才需主动 Rollback();否则可能掩盖真实问题。

实操建议:

  • 重点监控 OracleException.Number:遇到 1(SQL 错误)、2091(事务已回滚)、8176(一致性读失败)等,应立即 tx.Rollback()
  • ORA-00001(唯一约束)这类业务错误,可选择性处理:记录日志 + Rollback(),或捕获后转为用户提示而不回滚整个事务(视业务容忍度)
  • 网络中断(ORA-03113/03114)发生时,连接通常已断,再调 Rollback() 会抛新异常,此时应直接放弃事务并重建连接
  • 统一在 catch (OracleException ex) 块末尾加 if (tx != null && tx.Connection?.State == ConnectionState.Open) tx.Rollback();,比每个分支单独判断更稳妥

Oracle 事务的边界比表面看起来更“粘稠”:连接状态、保存点命名、异常分类,任何一个环节松动都会让回滚变成幻觉。最常被忽略的是连接存活时间——它不只影响性能,直接决定事务是否真的落地。

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
数据分析工具有哪些
数据分析工具有哪些

数据分析工具有Excel、SQL、Python、R、Tableau、Power BI、SAS、SPSS和MATLAB等。详细介绍:1、Excel,具有强大的计算和数据处理功能;2、SQL,可以进行数据查询、过滤、排序、聚合等操作;3、Python,拥有丰富的数据分析库;4、R,拥有丰富的统计分析库和图形库;5、Tableau,提供了直观易用的用户界面等等。

1135

2023.10.12

SQL中distinct的用法
SQL中distinct的用法

SQL中distinct的语法是“SELECT DISTINCT column1, column2,...,FROM table_name;”。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

340

2023.10.27

SQL中months_between使用方法
SQL中months_between使用方法

在SQL中,MONTHS_BETWEEN 是一个常见的函数,用于计算两个日期之间的月份差。想了解更多SQL的相关内容,可以阅读本专题下面的文章。

381

2024.02.23

SQL出现5120错误解决方法
SQL出现5120错误解决方法

SQL Server错误5120是由于没有足够的权限来访问或操作指定的数据库或文件引起的。想了解更多sql错误的相关内容,可以阅读本专题下面的文章。

2214

2024.03.06

sql procedure语法错误解决方法
sql procedure语法错误解决方法

sql procedure语法错误解决办法:1、仔细检查错误消息;2、检查语法规则;3、检查括号和引号;4、检查变量和参数;5、检查关键字和函数;6、逐步调试;7、参考文档和示例。想了解更多语法错误的相关内容,可以阅读本专题下面的文章。

380

2024.03.06

oracle数据库运行sql方法
oracle数据库运行sql方法

运行sql步骤包括:打开sql plus工具并连接到数据库。在提示符下输入sql语句。按enter键运行该语句。查看结果,错误消息或退出sql plus。想了解更多oracle数据库的相关内容,可以阅读本专题下面的文章。

1723

2024.04.07

sql中where的含义
sql中where的含义

sql中where子句用于从表中过滤数据,它基于指定条件选择特定的行。想了解更多where的相关内容,可以阅读本专题下面的文章。

586

2024.04.29

sql中删除表的语句是什么
sql中删除表的语句是什么

sql中用于删除表的语句是drop table。语法为drop table table_name;该语句将永久删除指定表的表和数据。想了解更多sql的相关内容,可以阅读本专题下面的文章。

441

2024.04.29

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

69

2026.03.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号