为DbContextScope添加数据库事务提交完成事件

使用EF开发应用程序的一个难点就在于对其DbContext的生命周期管理,你的管理策略是否能很好的支持上层服务 使用独立事务,使用嵌套事务,并行执行,异步执行等需求? Mehdi El Gueddari对此做了深入研究和优秀的工作并且写了一篇优秀的文章,为了方便更多的童鞋学习,我已将其翻译为中文系列 :
在EntityFramework6中管理DbContext的正确方式

当然,在使用Mehdi El Gueddari为我们提供的DbContextScope组件时,任然会遇到一些比较棘手的问题,比如:

using (var scope = _dbContextScopeFactory.Create())
{
    var db = scope.DbContexts.Get<RentalServiceDbContext>();

    var bookingInfo = db.BookingInfoes.Load(bookingId);

    //调用bookingInfo处理一些操作


    //当满足一定的条件,发布一个事件,事件的订阅者可以订阅该信息并处理
    if (bookingInfo.Status == BookingStatus.Ordered)
    {
        _roomEventSvc.PublishRoomEvent(bookingInfo.RoomId, bookingInfo.Id, RoomEventType.BookingRoom);
    }

    //提交数据库事务
    scope.SaveChanges();
}

有可能我们发布的事件会被很快订阅者处理,甚至在我们提交事务之前!但问题就在于事件的订阅者很可能假定我们已经提交事务了!

当然,我们很可能想到的最简单的处理问题的方法就是 将 发布事件的那个代码片段移到 提交 事务后面。确实可以这样,就这个Demo来说,这样处理简单方便。但我们可能面临更多复杂的场景,比如这个服务方法被嵌套在另外一个服务方法内部调用,而且 事务也自动升级为使用外部服务的 事务,那么这个时候上面的简单处理方法就失效了!

最好的办法就是我们能在 DbContextScope提交事务后,插入一段我们自己的代码逻辑来实现我们的业务逻辑功能,很显然,为DbContextScope增加一个提交事务事件即可,然后就像下面这样使用:

using (var scope = _dbContextScopeFactory.Create())
            {
                var db = scope.DbContexts.Get<RentalServiceDbContext>();

                var bookingInfo = db.BookingInfoes.Load(bookingId);

                //调用bookingInfo处理一些操作


                //当满足一定的条件,发布一个事件,事件的订阅者可以订阅该信息并处理
                if (bookingInfo.Status == BookingStatus.Ordered)
                {
                    //保证我们的PublishRoomEvent方法在事务提交后执行
                    scope.DbContexts.TransactionCommitted += (sender, args) =>
                    {
                        _roomEventSvc.PublishRoomEvent(bookingInfo.RoomId, bookingInfo.Id, RoomEventType.BookingRoom);
                    };
                }

                //提交数据库事务
                scope.SaveChanges();
            }

其实,只需要几行代码就可以为DbContextScope增加事务提交事件,在这儿就不具体描述了!

猜你喜欢

转载自blog.csdn.net/wulex/article/details/80489860