【问题标题】:How to handle unit of work with dependency injection and repositories如何使用依赖注入和存储库处理工作单元
【发布时间】:2013-02-25 21:37:52
【问题描述】:

我继承了一些具有 UnitOfWorkFactory 的代码,它在每个存储库方法中创建一个工作单元。问题是单个存储库方法很少是完整的工作单元,因此如果说 OrderService.PlaceOrder 出现问题,它不能只是回滚/丢弃该工作单元,因为它不是一个单元。

查看代码,我认为应该将工作单元移到服务类或演示者中。然后我遇到的问题是如何将它传递给服务或存储库?演示者获得了一个服务的实例,而服务获得了一个存储库的实例。

我可以创建工作单元并让它注入服务、存储库和演示者的构造函数,但它会超出单个工作单元。这是一个桌面应用程序,因此演示者及其传递的任何服务都可以用于多个工作单元。

我认为可以传递工作单元的唯一方法是将其作为参数添加到所有服务/存储库方法。我不禁想到肯定有比这更好的方法,我错过了什么吗?

代码如下所示:

存储库:

class OrderRepository
{
    public UnitOfWorkFactory UnitOfWorkFactory;

    public OrderRepository(UnitOfWorkFactory unitOfWorkFactory)
    {
        UnitOfWorkFactory = unitOfWorkFactory;
    }

    public void Save(Order order)
    {
        using(var uow = UnitOfWorkFactory.Create())
        {
            // save order
            uow.commit();
        }
    }
}

服务:

class OrderService
{
    protected IOrderRepository OrderRepository;
    protected IProductService ProductService;

    public OrderService(IOrderRepository orderRepository, IProductRepository productService)
    {
        OrderRepository = orderRepository;
        ProductService = productService;
    }

    public void PlaceOrder(Order order)
    {
        foreach(var item in order.Items)
        {
            if(!ProductService.IsInstock(item.Product, item.Quantity))
                throw new ProductOutOfStockException(product);

            ProductService.MarkForDispatch(item.Product, item.Quantity);
        }

        OrderRepository.Save(order);
    }

    public void CancelOrder(Order order)
    {
        ProductService.UnmarkForDispatch(item.Product, item.Quantity);

        order.IsCanceled = true;
        OrderRepository.Save(order);
    }
}

演讲者:

class OrderPresenter
{
    protected IOrderView OrderView;
    protected IOrderService OrderService;

    public OrderPresenter(IOrderView orderView, IOrderService orderService)
    {
        OrderView = orderView;
        OrderService = orderService;
    }

    public void PlaceOrder()
    {
        OrderService.PlaceOrder(order);
    }

    public void CanelOrder()
    {
        OrderService.CancelOrder(order);
    }
}

【问题讨论】:

  • 您使用的是胖客户端还是瘦客户端(如 HTTP)?
  • @jgauffin 这是一个胖客户端。这是一个桌面应用程序,可以在本地处理所有业务规则。

标签: design-patterns dependency-injection mvp unit-of-work


【解决方案1】:

对此我没有一个简短的答案。您可以在我之前写的关于这个主题的这篇博文中查看完整的答案。 [Yet Another UoW, Repository Article].
(如果您阅读了这篇文章,请告诉我您的想法)

【讨论】:

  • 这看起来是个不错的解决方案。我最终的做法略有不同(见下文),但我会将其标记为答案,因为它也可以解决问题。
【解决方案2】:

我最终决定使用工厂并将其传递给工作单元。然后,工厂使用传递的 UOW 创建其他所有内容。

如果最终这样做:

存储库:

public class Repository : IRepoository
{
    IUnitOfWork UnitOfWork;

    public Repository(IUnitOfWork uow)
    {
        UnitOfWork = uow;
    }

    public void SomeOtherMethod(int id)
    {
        // Do something
    }
}

服务:

public class Service : IService
{
    IUnitOfWork UnitOfWork;
    IRepoository Repository;

    public Service(IUnitOfWork uow, IRepoository repository)
    {
        UnitOfWork = uow;
        Repository = repository;
    }

    public void SomeMethod(int id)
    {
        Repository.SomeOtherMethod(id);
        // do other stuff
    }
}

工厂:

public class IUnitOfWorkFactory
{
    IUnitOfWork Create();
}

public class IServiceFactory
{
    IService Create(IUnitOfWork uow);
}

IServiceFactory.Create 方法创建服务和服务所依赖的存储库,并为它们提供传递的工作单元。

预置器:

public class Presenter
{
    protected UnitOfWorkFactory UnitOfWorkFactory;
    protected IServiceFactory ServiceFactory;

    public Presenter(IUnitOfWorkFactory unitOfWorkFactory, IServiceFactory serviceFactory)
    {
        UnitOfWorkFactory = unitOfWorkFactory;
        ServiceFactory = serviceFactory;
    }

    public void DoSomething()
    {
        // Get a new UOW from the factory
        using(var uow = unitOfWorkFactory.Create())
        {
            // Then create the service and repository with the new UOW via the ServiceFactory
            var service = serviceFactory.Create(uow);
            service.SomeMethod(20);

            uow.Commit();
        }
    }
}

【讨论】:

  • 这看起来也是一个不错的解决方案,但是您需要为每个服务创建一个工厂。如果我最终将你的一些想法添加到我的帖子中,我会记得给你信用。
  • 是的,这是这种方法的缺点。对我来说这不是问题,因为代码使用库进行依赖注入,因此我可以创建一个接受 UOW 的通用工厂接口,它会自动创建实现。您也可以手动创建一个通用工厂来轻松完成。
猜你喜欢
  • 2014-10-28
  • 2023-03-23
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多