NHibernate 如何对session管理,实现lazy=true

Nhibernate session管理。以前用过Hibernate,由于当时我不是主要负责持久层,所以对Hibernate不是很熟悉,但记得当时session管理没有什么问题。但是NHibernate就出现了一个问题。如果每次进行持久化操作都open一次session然后close一次session,那么将不能使用lazy这个机制。运行时会报错“session已关闭”之类的提示。 怎么解决呢?我查了一些文章得到以下结论。 两个方法: 1.自己写一个sessionFactoryHelper,里边建一个getCurrentSession方法,第一次就建一个session丢到HttpContext里边,基本不用关闭,等服务器自己销毁 http://www.developer.com/open/article.php/10930_3709346_4

2.在<httpModules>里边加入一个类,分别加入一个BeginRequest的Handler和EndRequest的Handler。使用的是Nhibernate对session绑定到request里边的机制。 http://hugh-lin.javaeye.com/blog/167730 本人在实践过程中,尝试了第二种方法。但使用Nhibernate的绑定时,出错,由于对NHibernate不熟悉,所以就完全不知道怎么解决。最后只能考虑结合第一个方法来创新一下。 最后解决过程如下: 1.建立一个NHibernateHelper这样的一个类,用于创建SessionFactory和创建session等工作。代码如下。其中实现IHttpModule接口是为了加入到Web初始化节点中,使得每次有request的时候,都会执行Application_BeginRequest和Application_EndRequest两个函数。一个是在request来的时候,打开session,放到上下文中,然后responce之前把session关闭。

using System;
using System.Web;
using NHibernate.Context;
using NHibernate;
using NHibernate.Cfg;
namespace DAL
{
    /// <summary>
    /// 主要用于管理session的Nhibernateclass
    /// </summary>
    public class NHibernateHelper : IHttpModule
    {
        private const string sessionKey = "nhibernate.current_session";
        private static readonly ISessionFactory sessionFactory;
        static readonly object padlock = new object();
        /// <summary>
        /// 初始化,建立只读的sessionfactory
        /// </summary>
        static NHibernateHelper()
        {
            sessionFactory = BuildSessionFactory("Model");              
        }
        private static ISessionFactory BuildSessionFactory(string AssemblyName)
        {
            NHibernate.Cfg.Configuration cfg = new Configuration();
            cfg.AddAssembly(AssemblyName);
            return cfg.BuildSessionFactory();
        }
        /// <summary>
        /// 初始化操作,在会话开始请求和会话结束请求事件处理中加上自定义的Application_BeginRequest和Application_EndRequest事件
        /// </summary>
        /// <param name="context"></param>
        public void Init(HttpApplication context)
        {
            context.BeginRequest += new EventHandler(Application_BeginRequest);
            context.EndRequest += new EventHandler(Application_EndRequest);
        }
        public void Dispose()
        {
        }
        /// <summary>
        /// 在一次会话请求开始的时候初始化当前的Httpcontext的session
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void Application_BeginRequest(object sender, EventArgs e)
        {
            HttpContext context = HttpContext.Current;
            ISession currentSession= sessionFactory.OpenSession();
            context.Items[sessionKey] = currentSession;
        }
        /// <summary>
        /// 在服务器端返回消息前,关闭当前httpcontext的session
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void Application_EndRequest(object sender, EventArgs e)
        {
            CloseSession();
        }
        /// <summary>
        /// 获取当前请求会话请求
        /// </summary>
        /// <returns></returns>
        public static ISession GetCurrentSession()
        {
            HttpContext context = HttpContext.Current;
            ISession currentSession = (ISession)context.Items[sessionKey];
            if (currentSession == null)
            {
                currentSession = sessionFactory.OpenSession();
                context.Items[sessionKey] = currentSession;
            }
            return currentSession;
        }
        /// <summary>
        /// 统一的关闭session操作
        /// </summary>
        public static void CloseSession()
        {
            HttpContext context = HttpContext.Current;
            ISession currentSession = (ISession)context.Items[sessionKey];
            if (currentSession == null)
            {
                // No current session
                return;
            }
            currentSession.Close();
            context.Items.Remove(sessionKey);
        }

    }
}

 2.要在Web.config中加入节点,使得web初始化的时候,初始化上边这个helper

<httpModules>
        <add name="NHibernateHelper" type="AssemblyName.NHibernateHelper" />
</httpModules>

3.使用的时候只需要调用helper,每次都getCurrentSession得到session。OK,完成……

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏恰同学骚年

自己动手写一个简单的MVC框架(第二版)

  在ASP.NET MVC中,最核心的当属“路由系统”,而路由系统的核心则源于一个强大的System.Web.Routing.dll组件。

762
来自专栏木宛城主

ASP.NET MVC 随想录——探索ASP.NET Identity 身份验证和基于角色的授权,中级篇

在前一篇文章中,我介绍了ASP.NET Identity 基本API的运用并创建了若干用户账号。那么在本篇文章中,我将继续ASP.NET Identity 之...

3966
来自专栏大内老A

WCF技术剖析之四:基于IIS的WCF服务寄宿(Hosting)实现揭秘

通过《再谈IIS与ASP.NET管道》的介绍,相信读者已经对IIS和ASP.NET的请求处理管道有了一个大致的了解,在此基础上去理解基于IIS服务寄宿的实现机制...

2039
来自专栏晓晨的专栏

Entity Framework Core 2.0 使用代码进行自动迁移

1333
来自专栏C#

ASP.NET SignalR2持久连接层解析

    越是到年底越是感觉浑身无力,看着啥也不想动,只期盼着年终奖的到来以此来给自己打一针强心剂。估摸着大多数人都跟我一样犯着这样浑身无力的病,感觉今年算是没挣...

2589
来自专栏小特工作室

基于微软企业库的AOP组件(含源码)

软件开发,离不开对日志的操作。日志可以帮助我们查找和检测问题,比较传统的日志是在方法执行前或后,手动调用日志代码保存。但自从AOP出现后,我们就可以避免这种繁琐...

2107
来自专栏菩提树下的杨过

.net中的认证(authentication)与授权(authorization)

注:这篇文章主要给新手看的,老手们可能会觉得没啥营养,就请绕过吧。 “认证”与“授权”是几乎所有系统中都会涉及的概念,通俗点讲: 认证(authenticat...

35710
来自专栏博客园

持久化方式

HttpContext抽象提供了一个简单的IDictionary<Object,Object>类型的字典集合,叫做Items。在每个请求中,这个集合从HttpR...

1152
来自专栏LanceToBigData

HttpClient(二)HttpClient使用Ip代理与处理连接超时

前言   其实前面写的那一点点东西都是轻轻点水,其实HttpClient还有很多强大的功能:   (1)实现了所有 HTTP 的方法(GET,POST,PUT,...

3358
来自专栏NetCore

【翻译】在Visual Studio中使用Asp.Net Core MVC创建你的第一个Web API应用(一)

HTTP is not just for serving up web pages. It’s also a powerful platform for bui...

2325

扫码关注云+社区