NHibernate投掷会话已关闭

时间:2013-05-20 19:49:55

标签: c# asp.net-mvc-3 nhibernate fluent-nhibernate ninject

我在风中飘扬,所以我想我会在这里问一下......如果这是显而易见的并且之前已经得到回答,请告诉我。

我正在构建一个MVC 3网站,当我与一个用户一起运行时,我点击了这些页面。但是,如果我疯狂地点击刷新,最终我会点击“会话已关闭”。

我几乎把我所有的存储库都隔离出去试图找到底部,所以我知道在主页上有错误。在存储库中调用的唯一内容是从数据库表中获取用户名。

我正在使用Postgres作为数据库,以及来自NauckIT的ASP.NET成员资格提供程序。主数据库也是Postgres(但另一个数据库)。

使用以下代码完成会话管理:

public class MvcApplication : System.Web.HttpApplication
{
    public static ISessionFactory SessionFactory = 
             NHibernateHelper.GetSessionFactory();

    public MvcApplication()
    {
        this.BeginRequest += MvcApplication_BeginRequest;
        this.EndRequest += MvcApplication_EndRequest;
    }

    void MvcApplication_BeginRequest(object sender, EventArgs e)
    {
        CurrentSessionContext.Bind(SessionFactory.OpenSession());
    }
    void MvcApplication_EndRequest(object sender, EventArgs e)
    {            
        CurrentSessionContext.Unbind(SessionFactory).Dispose();
    }
}

获取登录信息的代码是:

    public Login GetCurrentLogin()
    {
        return Session.Query<Login>().FirstOrDefault(l => l.UserID == UserAccessRepository.UserID);
    }

UserAccessRepository只是从表单身份验证Cookie中获取userid

使用以下命令将会话注入到存储库中

        ninjectKernel.Bind<IUserRepository>().To<NHUserRepository>();
        ninjectKernel.Bind<ILeagueRepository>().To<NHLeagueRepository>().InThreadScope();
        ninjectKernel.Bind<ISession>()
            .ToMethod(m => MvcApplication.SessionFactory.GetCurrentSession())

sessionfactory来自:

public class NHibernateHelper
{
    private static ISessionFactory _sessionFactory;

    public static ISessionFactory SessionFactory
    {
        get
        {
            if (_sessionFactory == null)
            {       var rawConfig = new Configuration();
                rawConfig.SetNamingStrategy(new PostgresNamingStrategy());
                var configuration = Fluently.Configure(rawConfig)
                    .Database(PostgreSQLConfiguration.Standard.ConnectionString(ConnectionString).ShowSql().Dialect("NHibernate.Dialect.PostgreSQL82Dialect"))
                    .Mappings(m =>
                                m.AutoMappings.Add(AutoMap.AssemblyOf<Event>(new AutoMapConfiguration())
                )).ExposeConfiguration(cfg => 
                    cfg.SetProperty("current_session_context_class", "web")
                _sessionFactory = configuration.BuildSessionFactory();
                Debug.WriteLine("Built SessionFactory");
            }
            return _sessionFactory;

要清楚,它在我点击页面的标准实例中工作正常,但是当我疯狂地点击F5时,我得到会话关闭的问题。

更新: 不确定它是否相关,但我在BaseController中从OnActionExecuting方法中看到了这个主要位置。它似乎已经在上述方法中得到了解决。

1 个答案:

答案 0 :(得分:1)

您不应该在网络应用中使用InThreadScope()。使用InRequestScope()。编辑阅读Object Scopes - 最近更新过,不知道它迟早会浪费你的时间!

如果您希望在会员提供商和请求处理中使用东西,则需要搜索Ninject Custom Provider(可能类似here)。