C#系列-EntityFrameworkCore.Transactions.Abstractions应用场景+实例(38)
- EntityFrameworkCore.Transactions.Abstractions应用场景
EntityFrameworkCore.Transactions.Abstractions 并不是一个官方的或广泛认可的 NuGet 包名称。在 Entity Framework Core (EF Core) 中,事务管理通常是通过 DbContext 的内置方法来实现的,如 SaveChanges 和 Database.BeginTransaction。然而,如果你想要实现事务管理的抽象层,这通常是为了提供以下应用场景中的灵活性或扩展性:
- 单元测试:
当你编写单元测试时,你可能想要抽象事务管理,以便能够更容易地控制事务的提交和回滚。通过创建一个事务抽象层,你可以在测试环境中模拟事务行为,比如模拟事务失败的情况。 - 跨多个数据库上下文的事务:
如果你的应用程序需要使用多个 DbContext 实例来操作不同的数据库或数据库架构,并且这些操作需要在单个事务中完成,那么你需要一个跨上下文的事务管理器。通过抽象事务管理,你可以确保所有数据库操作要么全部成功,要么全部失败。 - 依赖注入和可配置性:
使用依赖注入 (DI) 和配置系统,你可以将事务管理抽象为服务,并在运行时根据配置选择不同的事务策略。例如,你可能想要在开发环境中使用内存中的事务模拟,而在生产环境中使用数据库事务。 - 多数据库提供程序支持:
如果你的应用程序需要支持多种数据库提供程序(如 SQL Server、MySQL、PostgreSQL 等),则通过抽象事务管理,你可以编写与数据库无关的代码,并通过配置来切换不同的数据库提供程序。 - 扩展性和可维护性:
通过将事务管理抽象为独立的组件,你可以更容易地扩展和修改事务逻辑,而无需更改使用事务的代码。这有助于保持代码的清晰和可维护性。
为了实现事务管理的抽象层,你可以创建一个接口,该接口定义了事务开始、提交和回滚的方法,然后实现该接口的具体类,这些类使用 EF Core 的内置事务管理功能。下面是一个简单的事务管理抽象接口和实现的示例:
csharp代码
| public interface ITransactionManager | |
| { | |
| IDbContextTransaction BeginTransaction(); | |
| void CommitTransaction(IDbContextTransaction transaction); | |
| void RollbackTransaction(IDbContextTransaction transaction); | |
| } | |
| public class EfCoreTransactionManager : ITransactionManager | |
| { | |
| private readonly DbContext _dbContext; | |
| public EfCoreTransactionManager(DbContext dbContext) | |
| { | |
| _dbContext = dbContext; | |
| } | |
| public IDbContextTransaction BeginTransaction() | |
| { | |
| return _dbContext.Database.BeginTransaction(); | |
| } | |
| public void CommitTransaction(IDbContextTransaction transaction) | |
| { | |
| transaction.Commit(); | |
| } | |
| public void RollbackTransaction(IDbContextTransaction transaction) | |
| { | |
| transaction.Rollback(); | |
| } | |
| } |
然后,在你的服务或仓储类中,你可以注入 ITransactionManager 并使用它来管理事务:
csharp代码
| public class MyService | |
| { | |
| private readonly ITransactionManager _transactionManager; | |
| private readonly MyDbContext _context; | |
| public MyService(ITransactionManager transactionManager, MyDbContext context) | |
| { | |
| _transactionManager = transactionManager; | |
| _context = context; | |
| } | |
| public void PerformTransactionalWork() | |
| { | |
| var transaction = _transactionManager.BeginTransaction(); | |
| try | |
| { | |
| // 执行数据库操作 | |
| _context.Add(new MyEntity()); | |
| _context.SaveChanges(); | |
| // 如果一切正常,提交事务 | |
| _transactionManager.CommitTransaction(transaction); | |
| } | |
| catch | |
| { | |
| // 发生异常时回滚事务 | |
| _transactionManager.RollbackTransaction(transaction); | |
| throw; | |
| } | |
| } | |
| } |
在这个例子中,MyService 类不直接依赖于 DbContext 的事务管理功能,而是依赖于抽象的 ITransactionManager 接口。这提供了更大的灵活性和可测试性。
2.EntityFrameworkCore.Transactions.Abstractions应用实例
EntityFrameworkCore.Transactions.Abstractions 并不是一个实际存在的 NuGet 包。Entity Framework Core (EF Core) 本身提供了事务处理的功能,你可以通过 DbContext 的 Database 属性来访问事务相关的方法,如 BeginTransaction。
不过,如果你想了解如何在 EF Core 中使用事务抽象,你可能需要自定义一个事务管理器或者使用第三方库来提供事务抽象层。
下面是一个简单的 EF Core 事务使用示例,展示了如何在单个 DbContext 实例中使用事务:
csharp代码
| using Microsoft.EntityFrameworkCore; | |
| public class MyDbContext : DbContext | |
| { | |
| public DbSet<MyEntity> MyEntities { get; set; } | |
| // ... 其他DbSets和配置 ... | |
| public int SaveChangesAndCommitTransaction() | |
| { | |
| try | |
| { | |
| // 开始事务 | |
| using (var transaction = Database.BeginTransaction()) | |
| { | |
| try | |
| { | |
| // 添加或修改实体 | |
| var entity = new MyEntity { /* 初始化属性 */ }; | |
| MyEntities.Add(entity); | |
| // 保存更改 | |
| var result = SaveChanges(); | |
| // 如果需要,执行其他数据库操作... | |
| // 提交事务 | |
| transaction.Commit(); | |
| return result; | |
| } | |
| catch (Exception) | |
| { | |
| // 发生异常时回滚事务 | |
| transaction.Rollback(); | |
| throw; | |
| } | |
| } | |
| } | |
| catch | |
| { | |
| // 如果在事务块外部发生异常,则不处理 | |
| throw; | |
| } | |
| } | |
| } | |
| public class MyEntity | |
| { | |
| public int Id { get; set; } | |
| // ... 其他属性 ... | |
| } |
在上面的代码中,SaveChangesAndCommitTransaction 方法尝试在一个事务中执行一些数据库操作。如果在尝试添加或修改实体时发生任何异常,事务将回滚,以确保数据库的完整性。
如果你需要跨多个 DbContext 实例或多个数据库提供程序使用事务,你可能需要创建一个分布式事务管理器。这通常涉及到使用如 IDistributedTransaction 或其他事务管理API,如 TransactionScope。
例如,使用 TransactionScope 的一个简单示例:
csharp代码
| using System.Transactions; | |
| using Microsoft.EntityFrameworkCore; | |
| public void PerformTransactionalWork() | |
| { | |
| using (var scope = new TransactionScope()) | |
| { | |
| try | |
| { | |
| using (var context1 = new MyDbContext1()) | |
| { | |
| // 在context1中执行数据库操作 | |
| context1.SaveChanges(); | |
| } | |
| using (var context2 = new MyDbContext2()) | |
| { | |
| // 在context2中执行数据库操作 | |
| context2.SaveChanges(); | |
| } | |
| // 如果所有操作都成功,则提交事务 | |
| scope.Complete(); | |
| } | |
| catch | |
| { | |
| // 如果发生异常,则事务在离开using块时自动回滚 | |
| } | |
| } | |
| } |
在上面的 PerformTransactionalWork 方法中,我们创建了一个 TransactionScope,它管理一个事务,该事务跨越两个不同的 DbContext 实例。如果在这两个上下文中的任何一个中发生异常,事务都会回滚,确保数据的一致性。
请注意,使用分布式事务可能会影响性能,并可能需要在数据库服务器上进行特殊配置。因此,在决定使用它们之前,请仔细评估你的需求。
相关文章:
C#系列-EntityFrameworkCore.Transactions.Abstractions应用场景+实例(38)
EntityFrameworkCore.Transactions.Abstractions应用场景 EntityFrameworkCore.Transactions.Abstractions 并不是一个官方的或广泛认可的 NuGet 包名称。在 Entity Framework Core (EF Core) 中,事务管理通常是通过 DbContext 的内置方法来实现的,如 Sa…...
PMDG 737
在Simbrief中生成计划后下载两个文件 放到C:\Users\32497\AppData\Local\Packages\Microsoft.FlightSimulator_8wekyb3d8bbwe\LocalState\packages\pmdg-aircraft-737(微软商店版本) 加油 先在飞行计划中查看计划燃油数量 MCDU中, AIRPLANE SEVICE 第二页, REQUEST FUEL TR…...
深入探索Midjourney:领航人工智能的新征程
深入探索Midjourney:领航人工智能的新征程 引言 在这个数据驱动、以技术创新为核心的时代,Midjourney以其独特的特性在人工智能领域中崭露头角。作为一款前沿的人工智能工具,它不仅重新定义了人机交互的方式,而且为各行各业提供…...
ChatGPT高效提问—prompt实践(漏洞风险分析-重构建议-识别内存泄漏)
ChatGPT高效提问—prompt实践(漏洞风险分析-重构建议-识别内存泄漏) 1.1 漏洞和风险分析 ChatGPT还可以帮助开发人员预测代码的潜在风险,识别其中的安全漏洞,而不必先运行它,这可以让开发人员及早发现错误࿰…...
【AIGC】Stable Diffusion 的提示词入门
一、正向提示词和反向提示词 Stable Diffusion 中的提示词通常用于指导用户对生成的图像进行控制。这些提示词可以分为正向提示词(Positive Prompts)和反向提示词(Negative Prompts)两类,它们分别影响图像生成过程中的…...
力扣---通配符匹配
题目描述: 给你一个输入字符串 (s) 和一个字符模式 (p) ,请你实现一个支持 ? 和 * 匹配规则的通配符匹配: ? 可以匹配任何单个字符。 * 可以匹配任意字符序列(包括空字符序列)。 判定匹配成功的充要条件是ÿ…...
Rust 原生类型
提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 前言一、标量类型(scalar type)二、 复合类型(compound type)总结 前言 Rust 学习系列 ,rust中的原生类…...
09、全文检索 -- Solr -- SpringBoot 整合 Spring Data Solr (生成DAO组件 和 实现自定义查询方法)
目录 SpringBoot 整合 Spring Data SolrSpring Data Solr的功能(生成DAO组件):Spring Data Solr大致包括如下几方面功能:Query查询(属于半自动)代码演示:1、演示通过dao组件来保存文档1、实体类…...
C# CAD SelectionFilter下TypedValue数组
SelectionFilter是用于过滤AutoCAD实体的类,在AutoCAD中,可以使用它来选择具有特定属性的实体。构造SelectionFilter对象时,需要传入一个TypedValue数组,它用于定义选择规则。 在TypedValue数组中,每个元素表示一个选…...
python 爬虫篇(3)---->Beautiful Soup 网页解析库的使用(包含实例代码)
Beautiful Soup 网页解析库的使用 文章目录 Beautiful Soup 网页解析库的使用前言一、安装Beautiful Soup 和 lxml二、Beautiful Soup基本使用方法标签选择器1 .string --获取文本内容2 .name --获取标签本身名称3 .attrs[] --通过属性拿属性的值标准选择器find_all( name , at…...
第十二周学习报告
比赛 参加了一场 div 2 ,B 题,C 题没写出来,B 是一个排序去重+双指针,C题是要观察出一个数学结论(因为数据范围太大,我暴力做直接超时了) 排 6253 ,表现分是 998 &…...
Redis面试题整理(持续更新)
1. 缓存穿透? 缓存穿透是指查询一个一定不存在的数据,如果从存储层查不到数据则不写入缓存,这将导致这个不存在的数据每次请求都要到 DB 去查询,可能导致DB挂掉,这种情况大概率是遭到了攻击。 解决方案: …...
一周学会Django5 Python Web开发-Django5 Hello World编写
锋哥原创的Python Web开发 Django5视频教程: 2024版 Django5 Python web开发 视频教程(无废话版) 玩命更新中~_哔哩哔哩_bilibili2024版 Django5 Python web开发 视频教程(无废话版) 玩命更新中~共计14条视频,包括:2024版 Django5 Python we…...
讲解用Python处理Excel表格
我们今天来一起探索一下用Python怎么操作Excel文件。与word文件的操作库python-docx类似,Python也有专门的库为Excel文件的操作提供支持,这些库包括xlrd、xlwt、xlutils、openpyxl、xlsxwriter几种,其中我最喜欢用的是openpyxl,这…...
WEB APIs(1)
变量声明const(修饰常量) const优先,如react,基本const, 对于引用数据类型,可用const声明,因为储存的是地址 何为APIs 可以使用js操作HTML和浏览器 分类:DOM(文档对象…...
C++重新入门-基本输入输出
C 的 I/O 发生在流中,流是字节序列。如果字节流是从设备(如键盘、磁盘驱动器、网络连接等)流向内存,这叫做输入操作。如果字节流是从内存流向设备(如显示屏、打印机、磁盘驱动器、网络连接等),这…...
【C语言】解析刘谦春晚魔术《守岁共此时》
今年的春晚上刘谦表演了魔术《守岁共此时》,台上台下积极互动(尤其是小尼),十分的有趣。刘谦老师的魔术不仅仅是他的高超手法,还有这背后的严谨逻辑,下面我们来用C语言来解析魔术吧。 源代码 #define _CRT…...
剑指offer——数值的整数次方
目录 1. 题目描述2. 一般思路2.1 有问题的思路2.2 全面但不高效的思路2.3 面试小提示 3. 全面又高效的思路 1. 题目描述 题目:实现函数 double Power(double base,int exponent),求base 的exponent 次方。不得使用库函数,同时不需要考虑大数问题 2. 一般…...
Tied Block Convolution: 具有共享较薄滤波器的更简洁、更出色的CNN
摘要 https://arxiv.org/pdf/2009.12021.pdf 卷积是卷积神经网络(CNN)的主要构建块。我们观察到,随着通道数的增加,优化后的CNN通常具有高度相关的滤波器,这降低了特征表示的表达力。我们提出了Tied Block Convolutio…...
算法沉淀——BFS 解决 FloodFill 算法(leetcode真题剖析)
算法沉淀——BFS 解决 FloodFill 算法 01.图像渲染02.岛屿数量03.岛屿的最大面积04.被围绕的区域 BFS(广度优先搜索)解决 Flood Fill 算法的基本思想是通过从起始点开始,逐层向外扩展,访问所有与起始点相连且具有相同特性…...
网络编程(Modbus进阶)
思维导图 Modbus RTU(先学一点理论) 概念 Modbus RTU 是工业自动化领域 最广泛应用的串行通信协议,由 Modicon 公司(现施耐德电气)于 1979 年推出。它以 高效率、强健性、易实现的特点成为工业控制系统的通信标准。 包…...
Linux 文件类型,目录与路径,文件与目录管理
文件类型 后面的字符表示文件类型标志 普通文件:-(纯文本文件,二进制文件,数据格式文件) 如文本文件、图片、程序文件等。 目录文件:d(directory) 用来存放其他文件或子目录。 设备…...
IGP(Interior Gateway Protocol,内部网关协议)
IGP(Interior Gateway Protocol,内部网关协议) 是一种用于在一个自治系统(AS)内部传递路由信息的路由协议,主要用于在一个组织或机构的内部网络中决定数据包的最佳路径。与用于自治系统之间通信的 EGP&…...
什么是库存周转?如何用进销存系统提高库存周转率?
你可能听说过这样一句话: “利润不是赚出来的,是管出来的。” 尤其是在制造业、批发零售、电商这类“货堆成山”的行业,很多企业看着销售不错,账上却没钱、利润也不见了,一翻库存才发现: 一堆卖不动的旧货…...
解决本地部署 SmolVLM2 大语言模型运行 flash-attn 报错
出现的问题 安装 flash-attn 会一直卡在 build 那一步或者运行报错 解决办法 是因为你安装的 flash-attn 版本没有对应上,所以报错,到 https://github.com/Dao-AILab/flash-attention/releases 下载对应版本,cu、torch、cp 的版本一定要对…...
微信小程序云开发平台MySQL的连接方式
注:微信小程序云开发平台指的是腾讯云开发 先给结论:微信小程序云开发平台的MySQL,无法通过获取数据库连接信息的方式进行连接,连接只能通过云开发的SDK连接,具体要参考官方文档: 为什么? 因为…...
实现弹窗随键盘上移居中
实现弹窗随键盘上移的核心思路 在Android中,可以通过监听键盘的显示和隐藏事件,动态调整弹窗的位置。关键点在于获取键盘高度,并计算剩余屏幕空间以重新定位弹窗。 // 在Activity或Fragment中设置键盘监听 val rootView findViewById<V…...
高防服务器能够抵御哪些网络攻击呢?
高防服务器作为一种有着高度防御能力的服务器,可以帮助网站应对分布式拒绝服务攻击,有效识别和清理一些恶意的网络流量,为用户提供安全且稳定的网络环境,那么,高防服务器一般都可以抵御哪些网络攻击呢?下面…...
CVE-2020-17519源码分析与漏洞复现(Flink 任意文件读取)
漏洞概览 漏洞名称:Apache Flink REST API 任意文件读取漏洞CVE编号:CVE-2020-17519CVSS评分:7.5影响版本:Apache Flink 1.11.0、1.11.1、1.11.2修复版本:≥ 1.11.3 或 ≥ 1.12.0漏洞类型:路径遍历&#x…...
【Go语言基础【12】】指针:声明、取地址、解引用
文章目录 零、概述:指针 vs. 引用(类比其他语言)一、指针基础概念二、指针声明与初始化三、指针操作符1. &:取地址(拿到内存地址)2. *:解引用(拿到值) 四、空指针&am…...
