工作单元属于EF4,IoC(Unity)和存储库?

Jer*_*ose 7 entity-framework repository ioc-container unity-container unit-of-work

我看到几个与此有关的问题,但我仍然无法找到我正在寻找的答案,所以我发布了我的问题.如果另一个问题得到答案(我只是没有看到),请指出我.

我正在试图找出我的UnitOfWork所属的位置 - 特别是,在使用EF4和Unity与Repository模式时创建.

基本上,我有一个用于实现我的业务逻辑的服务.此服务构造函数接收存储库,因此服务将注入我的存储库.然后,该服务使用注入的存储库对数据存储执行操作 - 但我需要将它们包含在一个工作单元中.

但是,我的工作单元需要注入EF4上下文(或者,在我的情况下,和上下文的接口 - IObjectContext).而且我不确定UoW应该在哪里创建并注入上下文.

以下是我能想到的可能选项,其中没有一个看起来很理想:

  • 在服务构造函数中包含UoW,从而使用工作单元注入服务,然后使用我的EF4上下文注入.但这似乎是错误的,因为我不想在存储库的每个实例上创建我的UoW.

  • 使用container.Resolve进行按需创建以获取UoW的实例,注入我的EF4上下文.这似乎过于不得不经常点击IoC容器,而不是已经有权访问UoW.

  • 将上下文直接注入服务,允许我创建UoW(上下文).这看起来很糟糕,因为我现在已经将上下文暴露给服务,这应该被隔离到存储库.

所以我的问题是,这些方法中的一种是可以接受的,还是有其他方法我没想到?

提前致谢.

Lad*_*nka 12

可能有几种方法如何使用它,所以我将描述一个我认为有用的方法.

Imho定义UoW的地方是应用程序逻辑 - 调用业务层(业务服务)的逻辑.原因是UoW应该代表逻辑业务trasaction - 应用程序逻辑(或远程调用时的服务外观)定义什么是逻辑事务.因此,例如在MVC中,您可以使用架构,其中每个控制器操作代表单个UoW:

public class MyController : Controller
{
  public MyController(IFirstService firstService, ISecondService secondService,
    IUnitOfWork unitOfWork)
  { ... }

  [HttpPost]
  public ActionResult SomeAction(Model data)
  {
    _firstService.SomeProcessing(data);
    _secondService.SomeProcessing(data);
    _unitOfWork.SaveChanges();
    return RedirectToAction(...);
  }
}
Run Code Online (Sandbox Code Playgroud)

在这个例子中,我的控制器依赖于两个业务服务,并且操作同时调用它们 - 然后UoW然后保存两个服务执行的更改.这就是我认为UoW应该在控制器中可用的原因,因为如果你的应用层没有UoW的访问权限,你就不能从几个服务调用中编写(重用)你的逻辑(因为每个服务调用都可以调用它自己的SaveChanges).

其他方法是服务外观.Facade将是您的业务层的公共接口,它将隐藏服务组合:

_firstService.SomeProcessing(data);
_secondService.SomeProcessing(data);
_unitOfWork.SaveChanges();
Run Code Online (Sandbox Code Playgroud)

在这种情况下,UoW不会传递给控制器​​,而是传递给服务外观,服务外观将注入控制器.如果您的业务逻辑将通过Web服务(或其他远程技术)公开,您肯定会使用此方法.

您必须处理的最后一个问题是将UoW传递给服务.服务以及UoW被注入控制器(演示者,服务外观或其他),但同时必须将UoW(或ObjectContext)注入服务,以便内部使用的存储库可以使用它.为此,您需要正确的IoC生命周期管理器,以便它在同一"请求"中为所有注入返回相同的实例.在Web应用程序的情况下,您需要PerHttpRequest生命周期管理器(您必须由您自己实现,因为Unity不提供它).