概念

什么是延迟加载:所谓延迟加载就是当在真正需要数据的时候,才真正执行数据加载操作。可以简单理解为,只有在使用的时候,才会发出sql语句进行查询,数据是分N次读取。

什么是立即加载:所谓立即加载既是所有的相关数据一次被读取出来,而不是分N次。

一、延迟加载:NHibernate延迟加载须有Session存在,且配置为Lazy=true(默认)

DAL中编写方法如下:

//延迟加载

public Order LazyLoad()

{

//返回order对象后ISession将继续存在

return session.Get<Order>(2);

//返回order后ISession将会立即被销毁

//using (ISession iSession = new SessionManager().GetSession())

//{

//    Order order = iSession.Get<Order>(2);

//    return order;

//}

}

上面有两种返回order对象的方式,一种返回order后session继续存在,一种在返回order后session就被销毁了,我们先使用未被注释的方法,然后使用被注释了的代码。

DAL.Test中编写如下方法:

[Test]

public void LazyLoadTest()

{

Order order = sample.LazyLoad();

}

然后设置断点,进行测试调试,在即时窗口编写代码测试。

当返回order后session继续存在时,结果如下:

order.OrderDate.ToString()

"2009-10-17 11:22:46"

order.Customer.Firstname

"soldier"

order.Products.Count

2

这里可以看到,我们可以访问order对象的关联对象Customer和Products

当返回order后session立即被销毁时,结果如下:

order.OrderDate.ToString()

"2009-10-17 11:22:46"

order.Customer.Firstname

“order.Customer.Firstname”引发了“NHibernate.LazyInitializationException”类型的异常

base {NHibernate.HibernateException}: {"Initializing[Model.Entities.Customer#2]-Could not initialize proxy - no Session."}

EntityId: 2

EntityName: "Model.Entities.Customer"

order.Products.Count

“order.Products.Count”引发了“NHibernate.LazyInitializationException”类型的异常

base {NHibernate.HibernateException}: {"Initializing[Model.Entities.Order#2]-failed to lazily initialize a collection of role: Model.Entities.Order.Products, no session or session was closed"}

EntityId: 2

EntityName: "Model.Entities.Order"

这里我们可以访问order对象,这是理所当然的。但是不论我访问order的关联对象Customer还是Products时,都会报“NHibernate.LazyInitializationException”异常,因为访问时使用的是延迟加载,但是session此时已经被销毁了,所以报错,这说明要使用延迟加载就必须保证session是存在的。

二、立即加载:有三种方法进行立即加载,一、配置lazy=”false”;二、使用NHibernateUtil类;三、使用带fetchHQL语句

1)       配置lazy=”false”进行立即加载

首先修改order.hbm.xml文件,将

<bag name="Products" generic="true" table="OrderProduct">

修改为如下:

<bag name="Products" generic="true" table="OrderProduct" lazy="false">

然后再测试,结果如下:

order.OrderDate.ToString()

"2009-10-17 11:22:46"

order.Products.Count

2

order.Customer.Firstname

“order.Customer.Firstname”引发了“NHibernate.LazyInitializationException”类型的异常

base {NHibernate.HibernateException}: {"Initializing[Model.Entities.Customer#2]-Could not initialize proxy - no Session."}

EntityId: 2

EntityName: "Model.Entities.Customer"

这里可以看到配置了lazy="false"的Products可以访问,因为它的数据已经和order一起读取了,但是当我们访问Customer时,报出了上面我们曾经见到过的错误,因为在order的映射文件里并没有配置Customer为立即加载,所以报错。

2)       使用NHibernateUtil类进行强制立即加载

首先修改LazyLoad()函数的代码如下:

//返回order后ISession将会立即被销毁

using (ISession iSession = new SessionManager().GetSession())

{

Order order = iSession.Get<Order>(2);

NHibernateUtil.Initialize(order.Customer);                   //这里强制立即加载order的关联Customer对象

return order;

}

测试结果如下:

order.OrderDate.ToString()

"2009-10-17 11:22:46"

order.Products.Count

2

order.Customer.Firstname

"soldier"

哈哈,又可以访问Customer对象了。

3)       使用带fetch的HQL查询

a) 使用HQL查询方法也可以立即加载。HQL语句支持的连接类型为:inner join(内连接)、left outer join(左外连接)、right outer join(右外连接)、full join(全连接,不常用)。

b) “抓取fetch”连接允许仅仅使用一个选择语句就将相关联的对象随着他们的父对象的初始化而被初始化,可以有效的代替了映射文件中的外联接与延迟属性声明。

几点注意:

c) fetch不与setMaxResults() 或setFirstResult()共用,因为这些操作是基于结果集的,而在预先抓取集合时可能包含重复的数据,也就是说无法预先知道精确的行数。

d) fetch还不能与独立的with条件一起使用。通过在一次查询中fetch多个集合,可以制造出笛卡尔积,因此请多加注意。对多对多映射来说,同时join fetch多个集合角色可能在某些情况下给出并非预期的结果,也请小心。

e)使用full join fetch 与 right join fetch是没有意义的。 如果你使用属性级别的延迟获取,在第一个查询中可以使用 fetch all properties 来强制NHibernate立即取得那些原本需要延迟加载的属性。

删除order.hbm.xml中的lazy="false"属性设置,这样order的Products就不是立即加载了,并将LazyLoad()函数修改如下:

//延迟加载

public Order LazyLoad()

{

//返回order对象后ISession将继续存在

//return session.Get<Order>(2);

//返回order后ISession将会立即被销毁

//using (ISession iSession = new SessionManager().GetSession())

//{

//    Order order = iSession.Get<Order>(2);

//    NHibernateUtil.Initialize(order.Customer);

//    return order;

//}

//使用HQL的fetch进行立即加载

using (ISession iSession = new SessionManager().GetSession())

{

return iSession.CreateQuery("from Order o left join fetch o.Products where o.OrderId=2 ")

.UniqueResult<Order>();

}

}

调试测试结果如下:

order.OrderDate.ToString()

"2009-10-17 11:22:46"

order.Products.Count

2

order.Customer.Firstname

“order.Customer.Firstname”引发了“NHibernate.LazyInitializationException”类型的异常

base {NHibernate.HibernateException}: {"Initializing[Model.Entities.Customer#2]-Could not initialize proxy - no Session."}

EntityId: 2

EntityName: "Model.Entities.Customer"

在上面的HQL语句中通过fetch o.Products语句来立即加载Products,所以测试里可以访问Products,而访问Customer时,则出错了。

然后我们修改上面的HQL语句:

"from Order o left join fetch o.Products where o.OrderId=2 "

为如下

"from Order o left join fetch o.Products inner join fetch o.Customer where o.OrderId=2 "

并再进行同上面一样的测试看结果如何,我的如下:

order.OrderDate.ToString()

"2009-10-17 11:22:46"

order.Products.Count

2

order.Customer.Firstname

"soldier"

Hibernate 延迟加载和立即加载的更多相关文章

  1. Hibernate之加载策略(延迟加载与即时加载)和抓取策略(fetch)

    假设现在有Book和Category两张表,表的关系为双向的一对多,表结构如下: 假设现在我想查询id为2的那本书的书名,使用session.get(...)方法: Session session=H ...

  2. 解决hibernate中的懒加载(延迟加载)问题

    解决hibernate中的懒加载(延迟加载)问题   我们在开发的时候经常会遇到延迟加载问题,在实体映射时,多对一和多对多中,多的一样的属性默认是lazy="true"(即,默认是 ...

  3. mybatis和hibernate中的懒加载

    概念:所谓懒加载就是延时加载,延迟加载.什么时候用懒加载呢,我只能回答要用懒加载的时候就用懒加载.至于为什么要用懒加载呢,就是当我们要访问的数据量过大时,明显用缓存不太合适,因为内存容量有限 ,为了减 ...

  4. EF如何操作内存中的数据以及加载相关联表的数据:延迟加载、贪婪加载、显示加载

    之前的EF Code First系列讲了那么多如何配置实体和数据库表的关系,显然配置只是辅助,使用EF操作数据库才是每天开发中都需要用的,这个系列讲讲如何使用EF操作数据库.老版本的EF主要是通过Ob ...

  5. EF 延迟加载和预先加载

    最近悟出来一个道理,在这儿分享给大家:学历代表你的过去,能力代表你的现在,学习代表你的将来. 十年河东十年河西,莫欺少年穷 学无止境,精益求精   本节探讨延迟加载和预先加载 Entity Frame ...

  6. Entity Framework关联查询以及数据加载(延迟加载,预加载)

    数据加载分为延迟加载和预加载 EF的关联实体加载有三种方式:Lazy Loading,Eager Loading,Explicit Loading,其中Lazy Loading和Explicit Lo ...

  7. EF如何操作内存中的数据和加载外键数据:延迟加载、贪婪加载、显示加载

    EF如何操作内存中的数据和加载外键数据:延迟加载.贪婪加载.显示加载 之前的EF Code First系列讲了那么多如何配置实体和数据库表的关系,显然配置只是辅助,使用EF操作数据库才是每天开发中都需 ...

  8. 第六节: EF高级属性(二) 之延迟加载、立即加载、显示加载(含导航属性)

    一. 简介 上一个章节中,也介绍了立即加载和延迟加载,但上一个章节是针对单表而言的,不含外键,立即也好,延迟也好,都是指单表中的数据.但本章节重点介绍的三种加载方式均是针对含(导航属性.外键)的情况下 ...

  9. ASP.NET MVC深入浅出(被替换) 第一节: 结合EF的本地缓存属性来介绍【EF增删改操作】的几种形式 第三节: EF调用普通SQL语句的两类封装(ExecuteSqlCommand和SqlQuery ) 第四节: EF调用存储过程的通用写法和DBFirst模式子类调用的特有写法 第六节: EF高级属性(二) 之延迟加载、立即加载、显示加载(含导航属性) 第十节: EF的三种追踪

    ASP.NET MVC深入浅出(被替换)   一. 谈情怀-ASP.NET体系 从事.Net开发以来,最先接触的Web开发框架是Asp.Net WebForm,该框架高度封装,为了隐藏Http的无状态 ...

随机推荐

  1. 【NopCommerce源码架构学习-二】单例模式实现代码分析

    单例模式是是常用经典十几种设计模式中最简单的..NET中单例模式的实现也有很多种方式.下面我来介绍一下NopCommerce中单例模式实现. 我之前的文章就分析了一下nop中EngineContext ...

  2. 如何汉化XAF应用

    这是一个入门级的问题,应网友请求,总结一下XAF汉化过程的几个关键点. 一.所有Dev的控件的汉化,Dev官方有汉化文件.点击下载15.2版本. 正版用户登陆至官网是有专门的下载界面的,并且可以参与汉 ...

  3. android视频播放器

    RTSP(Real Time Streaming Protocol),RFC2326,实时流传输协议,是TCP/IP协议体系中的一个应用层协议,由哥伦比亚大学.网景和RealNetworks公司提交的 ...

  4. [连载]《C#通讯(串口和网络)框架的设计与实现》-4.设备驱动管理器的设计

    目       录 第四章           设备驱动管理器的设计... 2 4.1           接口定义... 2 4.2           设备容器... 7 4.3          ...

  5. VS 快捷键

    项目相关的快捷键 Ctrl + Shift + B = 生成项目 Ctrl + Alt + L = 显示Solution Explorer(解决方案资源管理器) Shift + Alt+ C = 添加 ...

  6. JS图片自动和可控的轮播切换特效

    点击这里查看效果:http://hovertree.com/texiao/js/1.htm HTML文件代码如下: <!DOCTYPE html> <html xmlns=" ...

  7. 码代码新神器-Github Atom

    周末闲着没事,逛论坛发现了一个新的编辑器,由github发布的Atom编辑器.瞬间被吸引了,所以就去尝试着折腾了一下,后来发现这个编辑器确实很不错,他的特点就是两个字:优美!!! 下载地址,官方网站下 ...

  8. gulp安装说明

    1.安装node-v6.3.0-x64,安装成功后再点击node-v6.3.0-x64卸载(点击remove). 2.安装node-v4.4.7-x64. 3.打开cmd命令行,输入node -v,查 ...

  9. SharePoint 2013 图文开发系列之自定义字段

    SharePoint使用的优势,就在于开箱即用.快速搭建,SharePoint自身为我们提供了很多字段类型,已经很丰富了.但是,在实际应用中,我们还需要一些功能特殊的字段,下面,我们简单介绍下字段的开 ...

  10. IBatis.Net项目数据库SqlServer迁移至Oracle经验

    最近完成了一个(IBatis.Net+MVC)项目的数据库+代码迁移工作,可把我折腾得~~~ IBatis.Net是一个ORM框架,具体介绍可以问度娘.我之前没用ORM框架使用经验,所以这一路我不是走 ...