【发布时间】:2013-08-21 11:48:18
【问题描述】:
我在 ASP.NET MVC 应用程序中将 Castle Windsor 与 NHIbernate 一起用于我的 IoC。注册后效果很好(有一个例外):
container.Register(Component.For<ISessionFactoryBuilder.().ImplementedBy<SessionFactoryBuilder>().LifestyleSingleton());
// Register the NHibernate session factory as a singleton using custom SessionFactoryBuilder.BuildSessionFactory method.
container.Register(Component.For<ISessionFactory>().UsingFactoryMethod(k => k.Resolve<ISessionFactoryBuilder>().BuildSessionFactory("ApplicationServices")).LifestyleSingleton());
container.Register(Component.For<IInterceptor>().ImplementedBy<ChangeAuditInfoInterceptor>().LifestylePerWebRequest());
container.Register(Component.For<ISession>().UsingFactoryMethod(k => k.Resolve<ISessionFactory>()
.OpenSession(container.Resolve<IInterceptor>())).LifestylePerWebRequest());
一切都很好,除了我的 ChangeAuditInterceptor 又注入了一个 IAccountSession 服务,该服务又注入了一个 NHibernate ISession...这导致以下循环依赖异常:
在尝试解析组件时检测到依赖循环 '后期绑定 NHibernate.ISession'。导致的解析树 周期如下:组件'后期绑定NHibernate.ISession' 解析为组件的依赖 'Blah.Core.Services.AccountSession' 解析为依赖 组件'Blah.Core.Infrastructure.Data.ChangeAuditInfoInterceptor' 解析为组件的依赖 'Blah.Core.Infrastructure.Installers.SessionFactoryBuilder' 已解决 作为组件“后期绑定 NHibernate.ISessionFactory”的依赖项 解析为组件“后期绑定 NHibernate.ISession”的依赖项 这是正在解析的根组件。
在过去的几年里,我通常使用 NHibernateSessionManager 运行,它负责插入 IInterceptor 而不会导致这种循环依赖问题(与使用 Castle Windsor 的 UsingFactoryMethod 功能的 SessionFactoryBuilder 的用法相反)。
关于如何解决这种循环依赖的任何建议?没有开始通过其他方式(即绕过问题并因此产生气味的属性注入)来破解 AccountSession 的 ISession。我已将 ISession 注入切换为 AccountSession 服务的属性注入,它工作正常,但我不喜欢隐式契约与构造函数显式契约。
public class AccountSession : IAccountSession
{
private readonly ISession _session;
public AccountSession(ISession session)
{
_session = session;
}
public Account GetCurrentAccount() // Called by a method in ChangeAuditInterceptor
{
...
}
...等等。
【问题讨论】:
-
我想你应该尝试将 _session 转换为公共自动属性(get;set;)并删除 AccountSession 的构造函数(因此为编译器自动生成的无参数公共构造函数留出位置) .值得一试(恕我直言)
-
这就是我所做的,它工作正常。但是,我不喜欢需要隐式契约和显式契约,尤其是在其他地方使用它时(尽管 DI 仍然由容器处理,所以你不必考虑它)。它闻起来很臭,因为它是一个周期性参考,我觉得我错过了一些东西(尽管不像典型的周期性参考那么脏)。
-
哦,感谢您对本期和上期的反馈。听起来你和我一样好奇什么是更好/最好的解决方案。我之前尝试过 Castle NHibernate Facility,但它需要 [Transaction] 标记来刷新和注入工厂等。也许答案是回到一个 NHibernate 会话管理器,它将被注入到任何地方(并且可以处理 IInterceptor注射本身)。多年来,我一直在使用 Ninject,最近转而尝试使用 Castle 和 Autofac。
标签: nhibernate castle-windsor isession