NHibernate教程(12)--延迟加载
本节内容
- 引入
- 延迟加载
- 实例分析
- 1.一对多关系实例
- 2.多对多关系实例
- 结语
引入
通过前面文章的分析,我们知道了如何使用NHibernate,比如CRUD操作、事务、一对多、多对多映射等问题,这篇我们初步探索NHibernate中的加载机制。
在讨论之前,我们看看我们使用的数据模型,回顾一下第二篇建立的数据模型。

Customer与Orders是一对多关系,Order与Product是多对多关系。这一篇还是使用这个模型,有关具体配置和映射参考本系列的文章。
延迟加载(Lazy Loading)
延迟加载按我现在的理解应该叫“视需要加载(load-on-demand)”,“(delayed loading)”,“刚好及时加载(just-in-time loading)”在合适不过了。这里按字面理解延迟仿佛变成了“延迟,延长,拖延时间”的意思。
NHibernate从1.2版本就默认支持了延迟加载。其实延迟加载的执行方式是使用GoF23中的代理模式,我们用一张图片来大致展示延迟加载机制。

实例分析
1.一对多关系实例
在一对多关系实例中,我们使用Customer对象与Order对象为例,在数据访问层中编写两个方法用于在测试时调用,分别是:
数据访问层中方法一:加载Customer对象
public Customer LazyLoad(int customerId)
{
return _session.Get<Customer>(customerId);
}
数据访问层中方法二:加载Customer对象并使用Using强制清理关闭Session
public Customer LazyLoadUsingSession(int customerId)
{
using (ISession _session = new SessionManager().GetSession())
{
return _session.Get<Customer>(customerId);
}
}
1.默认延迟加载
调用数据访问层中的LazyLoad方法加载一个Customer对象,NHibernate的默认延迟加载Customer关联的Order对象。利用NHibernate提供有用类(NHibernateUtil)测试被关联的Customer对象集合是否已初始化(也就是已加载)。
[Test]
public void LazyLoadTest()
{
Customer customer = _relation.LazyLoad(1);
Assert.IsFalse(NHibernateUtil.IsInitialized(customer.Orders));
}
测试成功,观察NHibernate生成SQL语句为一条查询Customer对象的语句。我们使用调试发现,Orders对象集合的属性值为:{Iesi.Collections.Generic.HashedSet`1[DomainModel.Entities.Order]},并可以同时看到Order对象集合中的项。截图如下:

2.延迟加载并关闭Session
同第一个测试相同,这个测试调用使用Using强制资源清理Session加载Customer对象的方法。
[Test]
public void LazyLoadUsingSessionTest()
{
Customer customer = _relation.LazyLoadUsingSession(1);
Assert.IsFalse(NHibernateUtil.IsInitialized(customer.Orders));
}
测试成功,其生成SQL语句和上面测试生成SQL语句相同。但是使用调试发现,Orders对象集合的属性值为:NHibernate.Collection.Generic.PersistentGenericSet<DomainModel.Entities.Order> ,如果你进一步想看看Order对象集合中的项,它抛出了HibernateException异常:failed to lazily initialize a collection, no session or session was closed。截图如下:

2.多对多关系实例
同理,在多对多关系实例中,我们以Order对象与Products对象为例,我们在数据访问层中写两个方法用于测试:
方法1:加载Order对象
public DomainModel.Entities.Order LazyLoadOrderAggregate(int orderId)
{
return _session.Get<DomainModel.Entities.Order>(orderId);
}
方法2:加载Customer对象并使用Using强制清理关闭Session
public DomainModel.Entities.Order LazyLoadOrderAggregateUsingSession(int orderId)
{
using (ISession _session = new SessionManager().GetSession())
{
return _session.Get<DomainModel.Entities.Order>(orderId);
}
}
1.默认延迟加载
调用数据访问层中的LazyLoadOrderAggregate方法加载一个Order对象,NHibernate的默认延迟加载Order关联的Products对象集合(多对多关系),利用代理模式加载Customer对象集合(多对一关系)。利用NHibernate提供有用类(NHibernateUtil)测试被关联的Products对象集合和Customer对象集合是否已初始化。
[Test]
public void LazyLoadOrderAggregateTest()
{
Order order = _relation.LazyLoadOrderAggregate(2);
Assert.IsFalse(NHibernateUtil.IsInitialized(order.Customer));
Assert.IsFalse(NHibernateUtil.IsInitialized(order.Products));
}
测试成功,NHibernate生成SQL语句如下:
SELECT order0_.OrderId as OrderId1_0_,
order0_.Version as Version1_0_,
order0_.OrderDate as OrderDate1_0_,
order0_.Customer as Customer1_0_
FROM [Order] order0_ WHERE order0_.OrderId=@p0; @p0 = '2'
调试看看效果截图,可以清楚的观察到Customer对象和Products对象集合的类型。

2.延迟加载并关闭Session
同第一个测试相同,这个测试调用使用Using强制资源清理Session加载Order对象的方法。
[Test]
public void LazyLoadOrderAggregateUsingSessionTest()
{
Order order = _relation.LazyLoadOrderAggregateUsingSession(2);
Assert.IsFalse(NHibernateUtil.IsInitialized(order.Customer));
Assert.IsFalse(NHibernateUtil.IsInitialized(order.Products));
}
测试成功,其生成SQL语句和上面测试生成SQL语句相同。但是使用调试发现,Customer对象类型为:{CustomerProxy9dfb54eca50247f69bfedd92e1638ba5},进一步观察Customer对象Firstname、Lastname等项引发了“NHibernate.LazyInitializationException”类型的异常。Products对象集合的属性值为:{NHibernate.Collection.Generic.PersistentGenericBag<DomainModel.Entities.Product>},如果你进一步想看看Products对象集合中的项它同样抛出HibernateException异常:failed to lazily initialize a collection, no session or session was closed。下面截取获取Customer对象一部分图,你想知道全部自己亲自调试一把:

3.延迟加载中LazyInitializationException异常
上面测试已经说明了这个问题:如果我想在Session清理关闭之后访问Order对象中的某些项会得到一个异常,由于session关闭,NHibernate不能为我们延迟加载Order项,我们编写一个测试方法验证一下:
[Test]
[ExpectedException(typeof(LazyInitializationException))]
public void LazyLoadOrderAggregateUsingSessionOnFailTest()
{
Order order = _relation.LazyLoadOrderAggregateUsingSession(2);
string name = order.Customer.Name.Fullname;
}
上面的测试抛出“Could not initialize proxy - no Session”预计的LazyInitializationException异常,表明测试成功,证明不能加载Order对象的Customer对象。
4.N+1选择问题
我们在加载Order后访问Product项,导致访问Product每项就会产生一个选择语句,我们用一个测试方法来模拟这种情况:
[Test]
public void LazyLoadOrderAggregateSelectBehaviorTest()
{
Order order = _relation.LazyLoadOrderAggregate(2);
float sum = 0.0F;
foreach (var item in order.Products)
{
sum += item.Cost;
}
Assert.AreEqual(21.0F, sum);
}
NHibernate生成SQL语句如下:
SELECT order0_.OrderId as OrderId1_0_,
order0_.Version as Version1_0_,
order0_.OrderDate as OrderDate1_0_,
order0_.Customer as Customer1_0_
FROM [Order] order0_ WHERE order0_.OrderId=@p0; @p0 = '2' SELECT products0_.[Order] as Order1_1_,
products0_.Product as Product1_,
product1_.ProductId as ProductId3_0_,
product1_.Version as Version3_0_,
product1_.Name as Name3_0_,
product1_.Cost as Cost3_0_
FROM OrderProduct products0_
left outer join Product product1_ on
products0_.Product=product1_.ProductId
WHERE products0_.[Order]=@p0; @p0 = '2'
这次我走运了,NHibernate自动生成最优化的查询语句,一口气加载了两个Product对象。但是试想一下有一个集合对象有100项,而你仅仅需要访问其中的一两项。这样加载所有项显然是资源的浪费。
幸好,NHibernate为这些问题有一个方案,它就是立即加载。欲知事后如何,请听下回分解!
结语
这篇我们初步认识了NHibernate中的加载机制,这篇从一对多关系、多对多关系角度分析了NHibernate默认加载行为——延迟加载,下篇继续分析立即加载。希望对你有所帮助
NHibernate教程(12)--延迟加载的更多相关文章
- NODE-WEBKIT教程(12)全屏
node-webkit教程(12)全屏 文/玄魂 目录 node-webkit教程(12)全屏 前言 12.1 和全屏有关的三个api Window.enterFullscreen() Window ...
- NHibernate教程
NHibernate教程 一.NHibernate简介 在今日的企业环境中,把面向对象的软件和关系数据库一起使用可能是相当麻烦.浪费时间的.NHibernate是一个面向.Net环境的对象/关系数据库 ...
- NHibernate教程(13)--立即加载
本节内容 引入 立即加载 实例分析 1.一对多关系实例 2.多对多关系实例 结语 引入 通过上一篇的介绍,我们知道了NHibernate中默认的加载机制--延迟加载.其本质就是使用GoF23中代理模式 ...
- [译]Vulkan教程(12)图形管道基础之入门
[译]Vulkan教程(12)图形管道基础之入门 Introduction 入门 Over the course of the next few chapters we'll be setting u ...
- [转帖]Linux教程(12)- linux输入输出重定向
Linux教程(12)- linux输入输出重定向 2018-08-21 22:57:02 钱婷婷 阅读数 49更多 分类专栏: Linux教程与操作 Linux教程与使用 版权声明:本文为博主原 ...
- 黑马lavarel教程---12、lavarel验证码
黑马lavarel教程---12.lavarel验证码 一.总结 一句话总结: 用插件的时候仔细看插件的版本要求 1.lavarel安装验证码插件的时候,如果(可选)需要定义自己的配置,则需要生成配置 ...
- Directx11教程(12) 禁止alt+enter全屏窗口
原文:Directx11教程(12) 禁止alt+enter全屏窗口 在D3D11应用程序中,我们按下alt+enter键,会切换到全屏模式.有时候,我们在WM_SIZE中有一些代码,全 ...
- 深度学习与CV教程(12) | 目标检测 (两阶段,R-CNN系列)
作者:韩信子@ShowMeAI 教程地址:http://www.showmeai.tech/tutorials/37 本文地址:http://www.showmeai.tech/article-det ...
- NHibernate教程(20)——二级缓存(上)
本节内容 引入 介绍NHibernate二级缓存 NHibernate二级缓存提供程序 实现NHibernate二级缓存 结语 引入 上一篇我介绍了NHibernate内置的一级缓存即ISession ...
随机推荐
- ASP.NET Core 源码学习之 Logging[3]:Logger
上一章,我们介绍了日志的配置,在熟悉了配置之后,自然是要了解一下在应用程序中如何使用,而本章则从最基本的使用开始,逐步去了解去源码. LoggerFactory 我们可以在构造函数中注入 ILogge ...
- 关于Python3中venv虚拟环境
Python3.3以上的版本通过venv模块原生支持虚拟环境,可以代替Python之前的virtualenv. 该venv模块提供了创建轻量级"虚拟环境",提供与系统Python的 ...
- Trailing Zeroes (III)
You task is to find minimal natural number N, so that N! contains exactly Q zeroes on the trail in d ...
- ASP.NET Core 快速入门【第二弹-实战篇】
上篇讲了asp.net core在linux上的环境部署.今天我们将做几个小玩意实战一下.用到的技术和工具有mysql.websocket.AngleSharp(爬虫html解析).nginx多站点部 ...
- ASP.NET WebApi 使用Swagger生成接口文档
前言 公司一直采用Word文档方式与客户端进行交流.随着时间的推移,接口变的越来越多,文档变得也很繁重.而且一份文档经常由多个开发人员维护,很难保证文档的完整性.而且有时写完代码也忘了去更新文档,为了 ...
- css 定位功能position
Static 定位 HTML元素的默认值,即没有定位,元素出现在正常的流中.静态定位的元素不会受到top, bottom, left, right影响. 相对定位Relative相对定位元素的定位是相 ...
- android学习笔记之GridView的使用
除了listview会使用适配器外,还有一种就是GridView,listview是单列多行的显示形式,适用于多项目的查看.而GridView是多行多列的显示形式,一般用在查看图片样式的activit ...
- MySQL里创建外键时错误的解决
--MySQL里创建外键时错误的解决 --------------------------------2014/04/30 在MySQL里创建外键时(Alter table xxx add const ...
- OpenGL与CUDA互操作方式总结
一.介绍 CUDA是Nvidia推出的一个通用GPU计算平台,对于提升并行任务的效率非常有帮助.本人主管的项目中采用了OpenGL做图像渲染,但是在数据处理方面比较慢,导致帧率一直上不来.于是就尝试把 ...
- Qt实现冒泡提示框
通过QLabel创建类似冒泡方式的提示框(提示框显示位置为父类控件居中位置,具体可根据需要自行修改),鼠标停留提示框界面时查看信息,离开时自动淡化消失的效果: 头文件定义 #ifndef _TTipW ...