抛弃TransactionScope
之前实现过类似功能是使用的TransactionScope,总碰到这样那样的问题,新项目迁移到.net core2.0下,果断抛弃之前的写法,因为DbContext的SaveChanges方法已经实现了UOW的功能
定义UOW
public interface IUnitOfWork { /// <summary> /// 提交事务 /// </summary> /// <returns>受影响行数</returns> int Commit(); }
public class UnitOfWork : IUnitOfWork { private readonly IDbContext _context; public UnitOfWork( IDbContext context) { _context = context ?? throw new ArgumentNullException(nameof(context)); } public int Commit() { return _context.SaveChanges(); } }
定义IActionFilter实现类
如果action执行结束后未发现异常,则提交事务(最终调用DbContext的SaveChanges方法)
public class GenericBusinessActionFilter : IActionFilter { private readonly IUnitOfWork _unitOfWork; public GenericBusinessActionFilter(IUnitOfWork unitOfWork) { _unitOfWork = unitOfWork; } public void OnActionExecuting(ActionExecutingContext context) { // do something before the action executes } public void OnActionExecuted(ActionExecutedContext context) { if (context.Exception == null) { //uow commit _unitOfWork.Commit(); } } }
注册过滤器
UOW注入
项目使用了autofac注入框架,未使用任何注入框架的同学可以通过.netcore自带的注入服务services.AddScoped方法实现.
services.AddScoped<IUnitOfWork, UnitOfWork>();
最关键的一点,引用圣杰的一句话:
确保Uow和Reopository之间共享同一个DbContext实例
这里我们限定了DbContext和UnitOfWork的生命周期为
Scoped
,从而确保每次请求共用同一个对象。如何理解呢?就是整个调用链上的需要注入的同类型对象,使用是同一个类型实例。
最后
本文实现了全局UOW,所有Action不用写任何代码就可以实现事务提交,看完一脸懵逼的同学需要先了解DDD和UOW相关概念
感谢『圣杰』的文章(http://www.cnblogs.com/sheng-jie/p/7416302.html#autoid-3-0-0)提供思路
时间: 2024-08-24 10:54:43