[NHibernate]延迟加载
目录
写在前面
上篇文章介绍了多对多关系的关联查询的sql,HQL,Criteria查询的三种方式。本篇文章将介绍nhibernate中的延迟加载方式,延迟加载按个人理解也可以叫做按需要加载(Loading-on-demand)。
文档与系列文章
[NHibernate]持久化类(Persistent Classes)
[NHibernate]集合类(Collections)映射
[NHibernate]缓存(NHibernate.Caches)
[NHibernate]NHibernate.Tool.hbm2net
[NHibernate]Nhibernate如何映射sqlserver中image字段
[NHibernate]条件查询Criteria Query
延迟加载
延迟加载(Lazy Load)是(也成为懒加载)Hibernate3关联关系对象默认的加载方式,延迟加载机制是为了避免一些无谓的性能开销而提出来的,所谓延迟加载就是当在真正需要数据的时候,才真正执行数据加载操作。可以简单理解为,只有在使用的时候,才会发出sql语句进行查询。
延迟加载的有效期是在session打开的情况下,当session关闭后,会报异常。当调用load方法加载对象时,返回代理对象,等到真正用到对象的内容时才发出sql语句。
Hibernate2实现延迟加载有2种方式:1实体对象,2集合。
Hibernate3中又引入了一种新的加载方式:3属性的延迟加载。
一般使用load的方法来实现延迟加载,在实现无限级联动使用延迟加载效率比较好。
——百度百科(java,hibernate)
上面是java中对hibernate的延迟加载的描述,说的比我好多了。
记住三点:1,为了避免无谓的性能开销。2,需要时才真正加载数据。3,使用了代理。
一个例子
一对多关系
默认延迟加载
采用懒加载的方式根据客户id得到客户信息
/// <summary>
/// 采用懒加载的方式根据客户id得到客户信息
/// </summary>
/// <param name="customerID"></param>
/// <returns></returns>
public Customer GetCustomerbyLazyLoad(Guid customerID)
{
//获得ISession实例
ISession session = NHibernateHelper.GetSession();
Customer customer = session.Get<Customer>(customerID);
return customer;
}
测试用代码,如图所示,此时数据已经加载出来了
使用SQL Profile监控生成的sql语句,截图如下
你会发现,此时只生成了查询customer的sql语句,nhibernate默认是使用延迟加载的,在前面的文章中,并没在映射文件中设置节点的lazy属性。
当展开customer的属性Orders时或者调试向下移动的时候,会执行查询,你会看到如下的sql语句
延迟加载并关闭Session
/// <summary>
/// 采用using释放session,懒加载的方式根据客户id得到客户信息
/// </summary>
/// <param name="customerID"></param>
/// <returns></returns>
public Customer GetCustomerbyLazyLoadUsing(Guid customerID)
{
//获得ISession实例
using(ISession session = NHibernateHelper.GetSession())
{
Customer customer = session.Get<Customer>(customerID);
return customer;
}
}
测试,当视图展开customer的orders属性,或者往下执行获得order集合时出错。
延迟加载,需要的时候再去加载,因为此时session已经关闭,没有去查询的通道了,结果是“此路不通”的提示。
多对多关系
默认延迟加载
这里采用多对多关系那篇文章中举的 Order和Product的例子。
/// <summary>
/// 多对多关系,延迟加载订单产品
/// </summary>
/// <returns></returns>
public Order GetOrderByLazyLoad(Guid orderId)
{
try
{
var session = NHibernateHelper.GetSession();
return session.Get<Wolfy.Shop.Domain.Entities.Order>(orderId);
}
catch (Exception)
{
throw;
}
}
进行测试,在查询的时候只有Order表的数据(这个地方,有Order数据的sql,猜测在一对多查询的时候,没有出现查询Customer的sql语句,很有可能是因为缓存的问题造成,因为经常使用Cusomer那条数据进行测试。)
展开Order的属性
当展开Order的属性,回去查询order下的所有的Product,此时生成的sql如下:
延迟加载并关闭Session
/// <summary>
/// 多对多关系,延迟加载订单产品
/// </summary>
/// <returns></returns>
public Order GetOrderByLazyLoadUsing(Guid orderId)
{
//session使用后即释放
using (var session = NHibernateHelper.GetSession())
{ return session.Get<Wolfy.Shop.Domain.Entities.Order>(orderId); }
}
测试
此时生成的sql语句
如果此时展开Products,同样会出现上面的异常
通过上面的比较一对多和多对多的默认延迟加载和关闭session后的情况类似。
N+1次select查询问题
如果Order下有很多个Product,而我们就想需要的时候才去加载其中某些Product的信息,如果采用立即加载的方式,势必产生多个sql语句,例如第一次查询得到所有的Order对象,然后根据orderid去查询得到所有的产品。
测试,多对多延迟加载订单和订单下单价大于6666的产品
protected void btnMany2Many_Click(object sender, EventArgs e)
{
Business.OrderBusiness orderBusiness = new Business.OrderBusiness();
Order order = orderBusiness.GetOrderByLazyLoad(new Guid("78A53F67-A293-48A1-BBE2-86FED77342FA"));
decimal sum = ;
foreach (var item in order.Products)
{
if (item.Price >= )
{
sum += item.Price;
}
}
Response.Write(sum.ToString());
}
生成的sql语句
延迟加载,可以解决select N+1的问题,比如你查询一个订单还有订单下的Product,势必会产生多条sql语句,先是查询order的sql语句,然后是查询product的sql语句,就会频繁的查询数据库,造成性能方面的压力,而采用延迟加载,通过上面生成的sql语句你会发现,使用left out join 将关联的表拼接起来,只生成了一条sql。
总结
本篇文章介绍了Nhibernate在延迟加载方面的内容,Nhibernate在使用过程中延迟加载方式是默认的。对延迟加载的定义,需要再慢慢的体会。
延迟加载(Lazy Load)是(也成为懒加载)Hibernate3关联关系对象默认的加载方式,延迟加载机制是为了避免一些无谓的性能开销而提出来的,所谓延迟加载就是当在真正需要数据的时候,才真正执行数据加载操作。可以简单理解为,只有在使用的时候,才会发出sql语句进行查询。
延迟加载的有效期是在session打开的情况下,当session关闭后,会报异常。当调用load方法加载对象时,返回代理对象,等到真正用到对象的内容时才发出sql语句。
Hibernate2实现延迟加载有2种方式:1实体对象,2集合。
Hibernate3中又引入了一种新的加载方式:3属性的延迟加载。
一般使用load的方法来实现延迟加载,在实现无限级联动使用延迟加载效率比较好。
——百度百科(java,hibernate)
[NHibernate]延迟加载的更多相关文章
- NHibernate 延迟加载与立即加载 (第七篇)
NHibernate 延迟加载与立即加载 (第七篇) 一.延迟加载 延迟加载可以理解为:当需要用的时候才加载. 假设我们数据库有一个Person对象,一个Country对象,其中Person属于Cou ...
- NHibernate系列文章二十一:延迟加载
摘要 NHibernate的延迟加载机制是很重要的内容.通过关系映射将数据库表之间的关系映射成对象之间的关系,如果没有延迟加载机制,从主表的一个对象的查询将直接查询出所有与该对象关联的其他对象,如果关 ...
- 耗时两月,NHibernate系列出炉
写在前面 这篇总结本来是昨天要写的,可昨天大学班长来视察工作,多喝了点,回来就倒头就睡了,也就把这篇总结的文章拖到了今天. nhibernate系列从开始着手写,到现在前后耗费大概两个月的时间,通过总 ...
- Hibernate 延迟加载和立即加载
概念 什么是延迟加载:所谓延迟加载就是当在真正需要数据的时候,才真正执行数据加载操作.可以简单理解为,只有在使用的时候,才会发出sql语句进行查询,数据是分N次读取. 什么是立即加载:所谓立即加载既是 ...
- [NHibernate]立即加载
目录 写在前面 文档与系列文章 立即加载 一个例子 总结 写在前面 上篇文章介绍了nhibernate延迟加载的相关内容,简单回顾一下延迟加载,就是需要的时候再去加载,需要的时候再向数据库发出sql指 ...
- [NHibernate]视图处理
目录 写在前面 文档与系列文章 视图 一个例子 总结 写在前面 前面的文章主要讲了对物理数据表的操作,当然了Nhibernate同样可以操作视图,本文将讲nhibernate对视图操作的种种. 文档与 ...
- [NHibernate]N+1 Select查询问题分析
目录 写在前面 文档与系列文章 N+1 Select查询问题分析 总结 写在前面 在前面的文章(延迟加载,立即加载)中都提到了N+1 Select的问题,总觉得理解的很不到位,也请大家原谅,这也是为什 ...
- [NHibernate]存储过程的使用(一)
目录 写在前面 文档与系列文章 Nhibernate中使用存储过程 一个例子 总结 写在前面 上篇文章一个小插曲,分析了延迟加载是如何解决N+1 select查询问题的.这篇开始介绍在nhiberna ...
- [NHibernate]存储过程的使用(二)
目录 写在前面 文档与系列文章 创建对象 更新对象 总结 写在前面 上篇文章介绍了如何使用MyGeneration代码生成器生成存储过程,以及nhibernate中通过存储过程删除数据的内容,这篇文章 ...
随机推荐
- android 开启本地相册选择图片并返回显示
.java package com.jerry.crop; import java.io.File; import android.app.Activity; import android.conte ...
- 理解 Node.js 里的 process.nextTick()
有很多人对Node.js里process.nextTick()的用法感到不理解,下面我们就来看一下process.nextTick()到底是什么,该如何使用. Node.js是单线程的,除了系统IO之 ...
- 无需FQ,自建本地CDN,秒上StackOverFlow!
StackOverflow是一个面向程序员的技术问答平台.可是在不FQ的情况下,浏览StackOverflow是一件让人极不舒服的事情,常常需要等待数十秒页面才慢慢显示出来.本文我教大家一种能够流畅地 ...
- Memcached在windows下安装与使用
建议:windows系统下仅为测试所有,生产环境下服务端应使用Linux系统. 本文最后更新于:2014-08-03 18:24 原文:http://www.yaosansi.com/post/mem ...
- -bash: /bin/rm: Argument list too long
使用rm * -f删除缓存目录文件时,报如下错误 -bash: /bin/rm: Argument list too long 提示文件数目太多. 解决的办法是使用如下命令: ls | xargs - ...
- [django]表格的添加与删除实例(可以借鉴参考)
自己并未采用任何表格插件,参考网上例子,自己编写出来的django网页实例,请各位参考! 首先看图做事,表格布局采用bootstrap,俗话说bootstrap橹多了就会css了,呵呵,下面看图: 上 ...
- windows下的NodeJS安装
1.登录官网 http://nodejs.org ,install 下载安装包.. 2.安装过程基本直接“NEXT”就可以了. 3.安装完成后可以使用cmd(win+r然后输入cmd进入)测试下是否安 ...
- java多线程系类:JUC线程池:04之线程池原理(三)(转)
转载请注明出处:http://www.cnblogs.com/skywang12345/p/3509960.html 本章介绍线程池的生命周期.在"Java多线程系列--"基础篇& ...
- 攻克Spring
http://www.cnblogs.com/dream-to-pku/p/5655247.html
- 树莓派Odroid等卡片式电脑上搭建NAS教程系列6-miniDLNA
目录: 1. 树莓派Odroid等卡片式电脑上搭建NAS教程系列1-Ubuntu系统安装 2. 树莓派Odroid等卡片式电脑上搭建NAS教程系列2-SSH连接访问 3. 树莓派Odroid等卡片式电 ...