摘要

NHibernate的延迟加载机制是很重要的内容。通过关系映射将数据库表之间的关系映射成对象之间的关系,如果没有延迟加载机制,从主表的一个对象的查询将直接查询出所有与该对象关联的其他对象,如果关联的对象上还有其他的关联对象,还要去查询其他的对象。如果这张“网”非常大,或者关联的数据库记录非常多的话,每次查询主表记录都要把整个数据库都查询一遍,这样效率会非常低下。为了解决这个问题产生了NHibernate延迟加载。对一些属性设置延迟加载,只在对象访问到这些属性的时候才去查询数据库,很大程度上提高了系统性能。

属性是否使用延迟加载在映射文件里设置lazy="true|false"来打开|关闭延迟加载。(NHibnerate默认是开启延迟加载,默认值是true,使用延迟加载)。

对于关联映射的属性,如果查询了主表数据,查询后程序中没有去读其关联属性,这时候如果将Session关闭,NHibernate就不能为我们实现延迟加载。换句话说,对于关联属性,延迟加载只在Session关闭之前有效。

程序演示

1、默认使用延迟加载的情况

假设数据库Customer表里有一条Id为2的记录,与之关联的Order表里有两条Order记录。

var customer = customerService.GetById();

执行此行代码得到的监控结果。

监控结果显示,只查询了Customer表记录。

如果将程序改成下面这样:

             var customer = customerService.GetById();
var orders = customer.Orders;
int orderCount = orders.Count;

再次运行程序,得到监控结果:

在执行这句下面赋值语句的时候,才去执行Order表的查询。

int orderCount = orders.Count;

2、关闭延迟加载

修改Customer.hbm.xml文件。

    <set name="Orders" table="`Order`" cascade="all-delete-orphan" inverse="true" lazy="false">
<key column="CustomerId"/>
<one-to-many class="Order"/>
</set>

执行同样的查询语句,得到监控结果。

监控记录显示,在查询Customer的同时,还查询了Order记录。

只有在关联的属性记录数量不是很多的情况下才使用lazy="false",关掉延迟加载。

 3、延迟加载和关闭Session

上面提到对于关联属性,延迟加载只在Session关闭之前有效。这里用程序演示一下。

修改Main函数。

 using Demo.Service;
using Demo.Service.Interface;
using System; namespace Demo.ConsoleApp
{
class Program
{
static readonly ICustomerService customerService = new CustomerService();
static readonly IOrderService orderService = new OrderService();
static readonly IProductService productService = new ProductService(); static void Main(string[] args)
{
HibernatingRhinos.Profiler.Appender.NHibernate.NHibernateProfiler.Initialize(); var customer = customerService.GetById();
Service.Infrastructure.SessionManager.CloseSession();
var orders = customer.Orders;
foreach (var order in orders)
{
Console.WriteLine(order.Id);
} Console.WriteLine("Completed");
Console.ReadLine();
}
}
}

执行程序,得到监控结果。

程序在执行for-each循环取Order对象的时候产生了异常。

如果删除这行代码程序能够正常运行。

Service.Infrastructure.SessionManager.CloseSession();

因此,为了能够利用对象的延迟加载,我们通常的做法是在查询完成后,不关闭Session。

有两种方法可以在关闭Session前,强制立即加载对象关联属性。

1)调用NHibernateUtil.Initialize静态方法,在Session关闭之前强制对指定的对象属性立即加载。

修改Main函数。

         static void Main(string[] args)
{
HibernatingRhinos.Profiler.Appender.NHibernate.NHibernateProfiler.Initialize(); var customer = customerService.GetById();
NHibernate.NHibernateUtil.Initialize(customer.Orders);
Service.Infrastructure.SessionManager.CloseSession(); var orders = customer.Orders;
foreach (var order in orders)
{
Console.WriteLine(order.Id);
} Console.WriteLine("Completed");
Console.ReadLine();
}

执行程序,得到监控结果。

2)在关联映射配置节中定义fetch="join",指出对于关联属性立即加载。

修改Customer.hbm.xml文件。

<?xml version="1.0" encoding="utf-8" ?>
<hibernate-mapping xmlns="urn:nhibernate-mapping-2.2" assembly="Demo.XML.Entities" namespace="Demo.XML.Entities.Domain">
<class name="Customer" table="Customer">
<id name="Id">
<generator class="native"/>
</id>
<property name="FirstName" not-null="true"/>
<property name="LastName" not-null ="true"/>
<property name="AverageRating"/>
<property name="Points"/>
<property name="HasGoldStatus"/>
<property name="MemberSince"/>
<property name="CreditRating" type="CustomerCreditRating"/>
<component name="Address">
<property name="Street"/>
<property name="City"/>
<property name="Province"/>
<property name="Country"/>
</component>
<set name="Orders" table="`Order`" cascade="all-delete-orphan" inverse="true" fetch="join">
<key column="CustomerId"/>
<one-to-many class="Order"/>
</set>
</class>
</hibernate-mapping>

修改Main函数。

         static void Main(string[] args)
{
HibernatingRhinos.Profiler.Appender.NHibernate.NHibernateProfiler.Initialize(); var customer = customerService.GetById();
Service.Infrastructure.SessionManager.CloseSession(); var orders = customer.Orders;
foreach (var order in orders)
{
Console.WriteLine(order.Id);
} Console.WriteLine("Completed");
Console.ReadLine();
}

执行程序,得到监控结果。

从监控结果看到,这次的查询SQL语句跟以前都不一样。之前执行了两条SQL语句,分别查询Customer表和Order表。这次只执行了一次SQL查询,从主表Customer到从表Order的left join查询。

有兴趣的可以用大数据测试的方法,比较哪种方法执行效率要高。

结语

NHibernate的延迟加载是NHibernate很重要的一个性能优化,NHibernate默认是开启延迟加载的,可以通过设置属性的lazy="false"强制关闭某一个属性的延迟加载。

NHibernate系列文章二十一:延迟加载的更多相关文章

  1. NHibernate系列文章二十四:NHibernate查询之Linq查询(附程序下载)

    摘要 NHibernate从3.0开始支持Linq查询.写Linq to NHibernate查询就跟写.net linq代码一样,非常灵活,可以很容易实现复杂的查询.这篇文章使用Linq to NH ...

  2. NHibernate系列文章二十八:NHibernate Mapping之Auto Mapping(附程序下载)

    摘要 上一篇文章介绍了Fluent NHibernate基础知识.但是,Fluent NHibernate提供了一种更方便的Mapping方法称为Auto Mapping.只需在代码中定义一些Conv ...

  3. NHibernate系列文章二十七:NHibernate Mapping之Fluent Mapping基础(附程序下载)

    摘要 从这一节起,介绍NHibernate Mapping的内容.前面文章都是使用的NHibernate XML Mapping.NHibernate XML Mapping是NHibernate最早 ...

  4. NHibernate系列文章二十五:NHibernate查询之Query Over查询(附程序下载)

    摘要 这一篇文章介绍在NHibernate 3.2里引入的Query Over查询,Query Over查询跟Criteria查询类似.首先创建IQueryOver对象,然后通过调用该对象的API函数 ...

  5. NHibernate系列文章二十三:NHibernate查询之Criteria查询(附程序下载)

    摘要 上一篇文章介绍了NHibernate HQL,他的缺点是不能够在编译时发现问题.如果数据库表结构有改动引起了实体关系映射的类有改动,要同时修改这些HQL字符串.这篇文章介绍NHibernate面 ...

  6. NHibernate系列文章二十二:NHibernate查询之HQL查询(附程序下载)

    摘要 NHibernate提供了多种查询方式,最早的HQL语言查询.Criteria查询和SQL Query,到NHibernate 3.0的Linq NHibernate,NHIbernate 4. ...

  7. NHibernate系列文章二十:NHibernate关系之一对一(附程序下载)

    摘要 NHibernate一对一关系虽然不经常碰到,但是在对于数据库结构优化的时候,经常会碰到一对一关系.比如,产品详细信息比较多的时候,可以把产品详细信息放到另一张表里面,Product主表只记录产 ...

  8. NHibernate系列文章二:创建NHibernate工程

    摘要 这篇文章介绍了如何创建一个简单的使用NHibernate的控制台应用程序,包括使用NuGet.简单的配置.单表映射.对NHibernate配置文件添加智能提示.使用ISessionFactory ...

  9. NHibernate系列文章二十六:NHibernate查询之SQL Query查询(附程序下载)

    摘要 NHibernate在很早的版本就提供了SQL Query(原生SQL查询),对于很复杂的查询,如果使用其他的查询方式实现比较困难的时候,一般使用SQL Query.使用SQL Query是基于 ...

随机推荐

  1. day26 分布式监控系统开发

    本节内容 为什么要做监控? 常用监控系统设计讨论 监控系统架构设计 监控表结构设计 为什么要做监控? –熟悉IT监控系统的设计原理 –开发一个简版的类Zabbix监控系统 –掌握自动化开发项目的程序设 ...

  2. SQL Server Profiler

    一.SQL Profiler工具简介 SQL Profiler是一个图形界面和一组系统存储过程,其作用如下: 图形化监视SQL Server查询: 在后台收集查询信息: 分析性能: 诊断像死锁之类的问 ...

  3. 通过 SMB 直通优化文件服务器的性能

    https://technet.microsoft.com/zh-cn/library/hh831487.aspx Windows Server 2012 内置新增功能,称为 SMB 直通,用来支持使 ...

  4. Oracle 用户管理与权限控制

    Oracle 用户管理与权限控制 oracle数据库的权限系统分为系统权限与对象权限.系统权限( database system privilege )可以让用户执行特定的命令集.例如,create ...

  5. Android ORMapping库

    自己用Java的注解实现了Android SQLite的ORM库,之前写过XML的,不过感觉不是很稳定,效率.鲁棒性各方面都不太好,今天花了一下午的时间,补全了所有的注解.注释,生成了javadoc, ...

  6. bug_ _

    java.lang.SecurityException: Not allowed to bind to service I app中加了百度定位功能,大部分手机测试没问题,但有部分手机会定位失败,提示 ...

  7. Linux文件权限概念

    一.Linux文件属性 1.第一列代表这个文件的类型与权限(permission): 共有10个字符 第一个字符代表这个文件的类型,是"目录,文件或链接文件等": [d]----& ...

  8. 从感知器到SVM

    这篇文章主要是分析感知器和SVM处理分类问题的原理,不涉及求解 感知器: 感知器要解决的是这样的一个二分类问题:给定了一个线性可分的数据集,我们需要找到一个超平面,将该数据集分开.这个超平面的描述如下 ...

  9. oledbdataadapter 读取excel数据时,有的单元格内容不能读出

    表现:excel中某列中,有的单元格左上角有绿色箭头标志,有的没有,c#编写读取程序,但是只能读取出带绿色箭头的单元格中的内容,其余不带的读取不到内容 原因:excel中单元格因为是文本格式而存储了数 ...

  10. 扁平设备树(FDT)

    组成 扁平设备树主要由4大部分组成:头部(header),预留内存块(memory reservation block),结构块(struct block)和字符串块(strings block).这 ...