zoukankan      html  css  js  c++  java
  • NHibernate Session-per-request and MiniProfiler.NHibernate

    NHibernate Session-per-request and MiniProfiler.NHibernate

    前言

      1、本文以mvc3为例,借鉴开源项目 NerdDnner项目完成nhibernate中的 Session-per-request 模式,本文创建了一个自定义的httpmodel类,来实现在http请求的时候创建并开启一个session并绑定到CurrentSessionContext中,当请求完成以后关闭,同时包含对事物的处理。

      2、利用MiniProfiler.NHibernate来追踪项目中的产生的sql,便于我们及时发现问题及时处理。MiniProfiler.NHibernate现在可以在nuget上直接获取或者可以去github中下载源码查看。

    实现Session per request

    复制代码
    public class NHibernateSessionPerRequest : IHttpModule
        {
            private static readonly ISessionFactory sessionFactory;
    
            //构造函数
            static NHibernateSessionPerRequest()
            {
                sessionFactory = CreateSessionFactory();
            }
    
            // 初始化httpmodel
            public void Init( HttpApplication context )
            {
                context.BeginRequest += BeginRequest;
                context.EndRequest += EndRequest;
            }
    
            public void Dispose() { }
    
            public static ISession GetCurrentSession()
            {
                return sessionFactory.GetCurrentSession();
            }
    
            // 打开session, 开启事务, 绑定session到CurrentSessionContext
            private static void BeginRequest( object sender, EventArgs e )
            {
                ISession session = sessionFactory.OpenSession();
    
                session.BeginTransaction();
    
                CurrentSessionContext.Bind( session );
            }
    
            // 移除session会话, 事物提交, and 关闭session会话
            private static void EndRequest( object sender, EventArgs e )
            {
                ISession session = CurrentSessionContext.Unbind( sessionFactory );
    
                if ( session == null ) return;
    
                try
                {
                    session.Transaction.Commit();
                }
                catch ( Exception )
                {
                    session.Transaction.Rollback();
                }
                finally
                {
                    session.Close();
                    session.Dispose();
                }
            }
    
            // 创建sessionfactory
            private static ISessionFactory CreateSessionFactory()
            {
                return Fluently.Configure(new Configuration().Configure())
                    .Mappings( m => m.AutoMappings.Add( CreateMappings() ) )
                    .ExposeConfiguration( UpdateSchema )
                    .CurrentSessionContext<WebSessionContext>()
                    .BuildSessionFactory();
            }
            
            private static AutoPersistenceModel CreateMappings()
            {
                return AutoMap
                    .Assembly( System.Reflection.Assembly.GetCallingAssembly() )
                    .Where( t => t.Namespace != null && t.Namespace.EndsWith( "Models" ) )
                    .Conventions.Setup( c => c.Add( DefaultCascade.SaveUpdate() ) );
            }
            
            // 生成数据库架构
            private static void UpdateSchema( Configuration cfg )
            {
                new SchemaUpdate( cfg )
                    .Execute( false, true );
            }
        }
    复制代码

    此类的nh的配置采用fluent的方式配置映射,可以生成数据库架构,ISessionFactory在每次请求中只会生成一次,当我们需要session的时候只需调用GetCurrentSession方法,当http请求的时候session创建并存储在CurrentSessionContext.Bind()中,并开启事物操作,当请求结束的时候 CurrentSessionContext.Unbind()移除session,事物提交并将session关闭。这里存在一些问题:尽管session是非常轻量级的这样每一次http请求都会去创建session,并不能做到我们真正需要的时候去创建。

    注册httpmodel:

    在web.config中添加如下2处节点:

    测试程序

    Models:

    复制代码
     View Code
    复制代码

    Repositories:

    复制代码
     View Code
    复制代码

    HomeController:

    复制代码
     View Code
    复制代码

    完成以后,修改hibernate.cfg.xml中的链接字符串等,并将其属性复制到输出目录修改为:如果较新则复制,运行程序请求index方法,将会产生数据库架构。

    配置MiniProfiler.NHibernate

    1、使用nuget控制台Install-Package MiniProfiler.NHibernate安装,或者参考github中的代码自己写一个

    2、修改NH配置文件中的数据库驱动类,将connection.driver_class几点替换为

    复制代码
    <property name="connection.driver_class">
          StackExchange.Profiling.NHibernate.Drivers.MiniProfilerSql2008ClientDriver,StackExchange.Profiling.NHibernate
        </property>
    复制代码

    3、在模板页中中的<head>节点添加@MiniProfiler.RenderIncludes()方法调用,修改Global.asax,添加如下代码:

    复制代码
    protected void Application_BeginRequest()
            {
                if (Request.IsLocal)
                {
                    MiniProfiler.Start();
                }
            }
    
            /// <summary>
            /// 终止时结束
            /// </summary>
            protected void Application_EndRequest()
            {
                MiniProfiler.Stop();
            }
    复制代码

    配置完成后运行程序,便可以看到MiniProfiler.NHibernate的效果了,使用它可以帮我们监控nh产生的sql及时优化代码,举个例子,运行请求home/seed的测试数据方法,看看追踪的效果

     

    文章结束,时间仓促代码粗略,文中若有不合理的地方,欢迎批评指正。

     
     
     
    标签: NHIBERNATE
  • 相关阅读:
    元组,字典
    for循环补充,变量和不可变量,数字类型,字符串类型,列表类型
    流程控制之while循环,for循环
    运算符,流程控制之if判断
    变量,常量,基本数据类型、运算符
    蓝桥杯--算法提高 排列数 (简单dfs)
    蓝桥杯-- 历届试题 核桃的数量 (gcd)
    hdoj--1272--小希的迷宫(并查集)
    zzulioj--1769--去师院的旅程:能怎么走(三)(0.0)
    zzulioj--1638--Happy Thanksgiving Day
  • 原文地址:https://www.cnblogs.com/Leo_wl/p/3273562.html
Copyright © 2011-2022 走看看