Closed. This question is
off-topic。它当前不接受答案。
想改善这个问题吗?
Update the question,所以它是
on-topic,用于堆栈溢出。
6年前关闭。
我正在使用:
-NHibernate 3.1版
-C#4.5
-温莎城堡2.5.3
NHibernate在几分钟内写入以下类型的错误:
[36]错误NHibernate.LazyInitializationException [(null)]-初始化[.Core。#23]-无法初始化代理-没有会话。
NHibernate.LazyInitializationException:正在初始化[.Core。#23]-无法初始化代理-没有会话。
正常情况下,一切正常,系统可以运行数天且在负载下运行等等。但是有时(特别是在发生某些数据库异常之后),它变得不稳定,并且在几分钟之内的许多请求过程中都无法恢复。 IIS重置“修复”它。
您能给我一些值得探讨的提示/可能的原因是什么?我该怎么做才能强制执行NHibernate重置或其他操作?
码:
private void ConfigureContainer()
{
_container = new DefaultTransientWindsorContainer();
_container.Register(Component.For<IRequestState>().ImplementedBy<PerHttpRequestState>(),
Component.For<ISessionFactory>()
.UsingFactoryMethod(x => InitNHibernate())
.LifeStyle.Singleton
);
_container.Install(FromAssembly.Containing<ZetesRegistry>());
_container.Install(FromAssembly.Containing<BusConnectorRegistry>());
Container.InitializeWith(_container);
}
更多代码:
internal class TransactionFilter : ActionFilterAttribute
{
public override void OnActionExecuting(System.Web.Http.Controllers.HttpActionContext actionContext)
{
var uow = UnitOfWork.Start();
System.Web.HttpContext.Current.Items["uow"] = uow;
base.OnActionExecuting(actionContext);
}
public override void OnActionExecuted(HttpActionExecutedContext actionExecutedContext)
{
if (actionExecutedContext.Exception == null)
{
var uow = System.Web.HttpContext.Current.Items["uow"] as UnitOfWork;
if (uow != null)
uow.Complete();
}
base.OnActionExecuted(actionExecutedContext);
}
}
提前谢谢!
更新1:
工作单元实施:
public class UnitOfWork : IUnitOfWork
{
public ISession Session { get; private set; }
public static IUnitOfWork Current
{
get { return Container.Resolve<IRequestState>().Get<IUnitOfWork>(); }
private set { Container.Resolve<IRequestState>().Store(value); }
}
private UnitOfWork(bool transactional)
{
var sessionFactory = Container.Resolve<ISessionFactory>();
Session = sessionFactory.OpenSession();
if (transactional) Session.BeginTransaction(IsolationLevel.ReadCommitted);
}
public static IUnitOfWork Start()
{
return Start(true);
}
private static IUnitOfWork Start(bool transactional)
{
if (Current != null)
throw new UnitOfWorkException("you cannot start more that one unit of work per request");
Current = new UnitOfWork(transactional);
return Current;
}
public static IUnitOfWork StartWithoutTransaction()
{
return Start(false);
}
public void Complete()
{
if (Session.Transaction.IsActive)
{
try
{
Session.Transaction.Commit();
}
catch
{
Session.Transaction.Rollback();
throw;
}
finally
{
Session.Close();
}
}
else
{
Session.Flush();
}
}
public void Dispose()
{
if (Session != null)
{
var transaction = Session.Transaction;
if (transaction != null
&& transaction.IsActive
&& !transaction.WasCommitted
&& !transaction.WasRolledBack)
{
transaction.Rollback();
}
Session.Dispose();
}
Current = null;
}
}
引发错误的代码:
public class ShippingsController : ApiController
{
private readonly IRepository _repository;
private readonly IPlanningDateProvider _planningDateProvider;
private readonly IStrategicPlanningProvider _strategicPlanningProvider;
public ShippingsController(IRepository repository, IPlanningDateProvider planningDateProvider, IStrategicPlanningProvider strategicPlanningProvider)
{
_repository = repository;
_planningDateProvider = planningDateProvider;
_strategicPlanningProvider = strategicPlanningProvider;
}
public Shippings Get(int id)
{
return Get(id, null);
}
public Shippings Get(int id, int? today)
{
var operationalPlanningCutOff = System.Web.Configuration.WebConfigurationManager.AppSettings["OperationalPlanningCutOff"];
if (string.IsNullOrEmpty(operationalPlanningCutOff))
operationalPlanningCutOff = "5";
var operationalPlanningCutOffHour = int.Parse(operationalPlanningCutOff);
if (today.HasValue && today.Value == 1)
return _repository.Query(new GetUrgentShippingInfoForCity(id)).UniqueResult();
return _repository.Query(new GetShippingInfoForCity(id, _planningDateProvider, operationalPlanningCutOffHour, _strategicPlanningProvider)).UniqueResult();
}
}
不知道这是否相关,但是我注意到我需要为未击中onResultExecuted
的过滤器异常添加回滚。记不清为什么需要这样做了,但是在某些情况下,我注意到我的数据库可能留有一个已启动但从未提交/回滚的事务,因为onResultExecuted
从未被命中,因此在以后的访问中导致数据库连接问题。也许您正在遇到这种情况。您将需要对其进行更改以适合您的UOW模式。
public void OnException(ExceptionContext filterContext)
{
if (CurrentSessionContext.HasBind(SessionFactory))
{
var currentTransaction = CurrentSessionContext.Unbind(SessionFactory).Transaction;
try
{
if (currentTransaction.IsActive)
currentTransaction.Rollback();
}
finally
{
currentTransaction.Dispose();
}
}
}
我是一名优秀的程序员,十分优秀!