我看到了几个与此有关的问题,但我仍然找不到我想要的答案,所以我正在张贴我的问题。如果另一个问题有答案(我只是没看到),请给我指点。
我试图找出我的UnitOfWork属于哪里--特别是创建--当使用EF4和模式时。
基本上,我有一个用于实现业务逻辑的服务。这个服务构造函数接受存储库,因此服务被注入到我的存储库中。然后,服务使用注入的存储库对数据存储执行操作--但我需要将这些操作封装在一个工作单元中。
然而,我的工作单元需要注入EF4上下文(或者,在我的例子中,是上下文-IObjectContext的接口)。我不知道应该在哪里创建和注入UoW /上下文。
以下是我能想到的可能的选择,其中没有一个看起来是理想的:
所以我的问题是,这些方法中的一种是可以接受的,还是我没有想到的另一种方法?
提前谢谢。
发布于 2011-02-17 09:00:43
可能有几种方法来使用它,所以我将描述一种我认为有用的方法。
定义UoW的地方是应用程序逻辑--调用业务层(业务服务)的逻辑。原因是UoW应该表示逻辑业务事务--应用程序逻辑(或远程调用情况下的服务外观)定义了什么是逻辑事务。因此,例如,在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(...);
}
}在本例中,我的控制器依赖于两个业务服务,并且操作都调用它们- UoW,然后保存由两个服务执行的更改。这就是为什么我认为UoW应该在控制器中可用的原因,因为如果您的应用层无法访问UoW,您就无法从几个服务调用中组合(重用)您的逻辑(因为每个调用都可能调用自己的SaveChanges)。
另一种方法是服务外观。Facade将是您业务层的公共接口,它将隐藏服务组合:
_firstService.SomeProcessing(data);
_secondService.SomeProcessing(data);
_unitOfWork.SaveChanges();在这种情况下,UoW不会传递给控制器,而是传递给服务外观,服务外观将被注入控制器。如果您的业务逻辑将通过web服务(或其他远程技术)公开,您肯定会使用这种方法。
您必须处理的最后一个问题是将UoW传递给服务。将服务和UoW注入控制器(演示器、服务外观或其他任何东西),但同时必须将UoW (或ObjectContext)注入服务,以便内部使用的存储库能够使用它。为此,您需要正确的IoC生存期管理器,以便它为同一“请求”内的所有注入返回相同的实例。在web应用程序的情况下,您需要PerHttpRequest生命周期管理器(您必须自己实现它,因为团结没有提供它)。
发布于 2011-02-17 18:31:09
一种管理方法是使用http://mfelicio.wordpress.com/2010/02/07/managing-the-entity-framework-objectcontext-instance-lifetime-in-wcf-and-sharing-it-among-repositories/中描述的方法,本文为Wcf服务实现了ContextManager。对于ASP.NET应用程序,我们可以使用类似的东西。
public class AspNetDBContextManager<TContext> : IDBContextManager
where TContext : IDBContext, new()
{
#region IDBContextManager Members
public IDBContext GetDBContext()
{
return this.GetOrCreateDbContext();
}
private IDBContext GetOrCreateDbContext()
{
if (HttpContext.Current == null)
{
throw new InvalidOperationException("Can be used only within ASP.NET applications");
}
string dbContextKey = string.Format("__AspNetDBCM__{0}__", HttpContext.Current.GetHashCode());
object dbContext = HttpContext.Current.Items[dbContextKey];
if (dbContext == null)
{
dbContext = new TContext();
if (dbContext != null)
{
HttpContext.Current.Items[dbContextKey] = dbContext;
}
}
return dbContext as IDBContext;
}
#endregion
}
public interface IDBContext
{
object Context { get; }
}
public interface IDBContextManager
{
IDBContext GetDBContext();
}https://stackoverflow.com/questions/5026265
复制相似问题