这篇文章

http://ayende.com/blog/3988/nhibernate-the-difference-between-get-load-and-querying-by-id

One of the more common mistakes that I see people doing with NHibernate is related to how they are loading entities by the primary key. This is because there are important differences between the three options.

The most common mistake that I see is using a query to load by id. in particular when using Linq for NHibernate.

var customer = (
select customer from s.Linq<Customer>()
where customer.Id = customerId
select customer
).FirstOrDefault();
Every time that I see something like that, I wince a little inside. The reason for that is quite simple. This is doing a query by primary key. The key word here is a query. This means that we have to hit the database in order to get a result for this query. Unless you are using the query cache (which by default you won’t), this force a query on the database, bypassing both the first level identity map and the second level cache. Get and Load are here for a reason, they provide a way to get an entity by primary key. That is important for several aspects, most importantly, it means that NHibernate can apply quite a few optimizations for this process.

好像在说,不要用主键(by primary key)去查询,因为用主键区查询会绕开一级和二级缓存。

所以建议用Get和Load去查询

下面的测试是错误的,要看正确的盖棺定论,请跳转至:
终极测试:

--------------------------------------------

实现延迟加载:

如何实现延迟加载:

要在实现延迟加载的类的映射文件xxx.hbm.xml中添加  lazy="true",根据加的位置,有两种:

一是类级别的:

  <class name="Model.Customer, Model"
table="Customer"
discriminator-value="0" lazy="true">
<!--unsaved-value="0" 主键表中不需要定义,而是需要在子表中定义-->
<id name="CustomerId"
column="CustomerId"
type="int"
unsaved-value="0">
<generator class="native" />
。。。。

二是集合级别的

  <class name="Model.Customer, Model"
table="Customer"
discriminator-value="0">
。。。。。
<set name="Orders" table="Order" lazy="true"
generic="true"
inverse="true" cascade="all">
<key column="CustomerId" foreign-key="FK_CustomerOrders"/>
<one-to-many class="Model.Order,Model"/>
</set>
。。。

根据目前的测试,集合位置的延迟加载配置会覆盖类级别的延迟加载配置。

有两种获取实体的方法:Get和Load方法。测试了如下组合:

类级别的lazy="true"  且                 类级别的lazy="true"且

集合级别的lazy="true"                  集合级别的lazy="false"

--------------------------------------------------------------------------

Get方法     |        加载父,延迟加载子                     父子都非延迟加载

Load方法   |        父子都非延迟加载                        父子都非延迟加载

从上面可以看到:

Load方法是不理会lazy="true" 配置的,永远的是非延迟加载。用Lode方法获取Customer时,Customer和子集合Order永远都是立即加载(不管哪里配置了lazy =“true”)。

刚好和:

http://blog.knowsky.com/192505.htm

描述的完全相反。到底谁对谁错,纠结啊

通过一夜的调试:

发现,如果是设断点去调试,那么Load表现为非延迟加载,发送了SQL语句,

这可能是编译器的问题。上面说的

刚好和:

http://blog.knowsky.com/192505.htm

描述的完全相反。到底谁对谁错,纠结啊


应该可以更正回来了,

类级别的lazy="true"  且                 类级别的lazy="true"且

集合级别的lazy="true"                  集合级别的lazy="false"

--------------------------------------------------------------------------

Get方法     |        加载父,延迟加载子                     父子都非延迟加载

Load方法   |        父子都延迟加载                          父子都非延迟加载

最终的结论是:

在配置lazy = “true”的情况下,

Get方法仅延迟加载子集合Orders,

Load方法比Get更牛逼,不仅延迟加载子集合Orders,把父Customer也延迟加载了。

Load方法是假设父Customer是存在,若不存在,会引发异常“NHibernate.ObjectNotFoundException”类型的异常,

但是这种异常,Nhibernate自己内部处理了,并返回一个实体类的代理(CustmoeProxy),保证不返回一个null

如下图所示:

------------------------------------------

非延迟加载图解:

延迟加载图解:

看到,当实现延迟加载,当需要Orders时,会通过session去查询,session。但此时,由于session已经关闭,

所以会抛出异常:

{"Initializing[Model.Customer#336]-failed to lazily initialize a collection of role: Model.Customer.Orders, no session or session was closed"}
为了避免session被关闭,延迟加载失败。有如下解决方案:
public Orders LazyLoadOrders(int customerId)
{
using (ISession _session = new SessionManager().GetSession())
{
return _session.Get<Customer>(customerId).Orders;
}
}

以后就调用这个方法来获取Orders

--------------------------------------------------------------------------------------------------------------
终极测试: 注意:测试时不要打断点,否则会影响测试结果(如果打了断点调试,会有多余的SQl请求产生),上面的测试就是因为打了断点去测试,结果
影响了测试结论。 测试的最终结论,上面所以的测试结论以下面的测试结果为盖棺定论。 下面的<class>和<Set>都是属于Customer.hbm.xml中的设置。

【1】Get方法:

<class  name ="Customer" lazy="true" >     <class  name ="Customer" lazy="false" >

<set name="Orders" lazy="true">        加载Customer,不加载Orders                         加载Customer,不加载Orders

<set name="Orders" lazy="false">       加载Customer,   加载Orders                         加载Customer,    加载Orders

    结论:1.Get方法永远加载Customer;
2.Get方法是否加载Orders,由<set name="Orders" lazy="">决定。  【2】Load方法:

<class  name ="Customer" lazy="true" >     <class  name ="Customer" lazy="false" >

<set name="Orders" lazy="true">        Customer和Orders都不加载                          加载Customer,不加载Orders

<set name="Orders" lazy="false">       Customer和Orders都不加载                          加载Customer,加载Orders

    结论:1.Load方法只要Class的Lazy为false,永远都不加载Customer和Orders;
2.Load方法只要Class的Lazy为ture,永远加载Customer,是否加载Oders由<set name="Orders" lazy="">决定。 

01-05-01-1【Nhibernate (版本3.3.1.4000) 出入江湖】延迟加载及其class和集合(set、bag等)的Lazy属性配置组合对Get和Load方法的影响的更多相关文章

  1. 01-08-05【Nhibernate (版本3.3.1.4000) 出入江湖】NHibernate二级缓存:第三方MemCache缓存

    一.准备工作 [1]根据操作系统(位数)选择下载相应版本的MemCache, MemCache的下载和安装,参看: http://www.cnblogs.com/easy5weikai/p/37606 ...

  2. 01-03-02-2【Nhibernate (版本3.3.1.4000) 出入江湖】CRUP操作-Save方法的一些问题

    此文由于当时不知道NHibernate的Sava方法不是更新操作,不知道Save就是Add,造成如下荒唐的求证过程,但结论是对的 ,可报废此文,特此声明. NHibernate--Save方法: Cu ...

  3. 01-08-01【Nhibernate (版本3.3.1.4000) 出入江湖】NHibernate中的一级缓存

    缓存的范围? 1.事务范围 事务范围的缓存只能被当前事务访问,每个事务都有各自的缓存,缓存内的数据通常采用相互关联的对象形式.缓存的生命周期依赖于事务的生命周期,只有当事务结束时,缓存的生命周期才会结 ...

  4. 01-04-03【Nhibernate (版本3.3.1.4000) 出入江湖】Criteria API关联查询

    Criteria API关联查询 如果说HQL查询还有需要了解点SQL语法知识,并不是完全彻底面向对象查询, 那么Criterial API就是完全面向对象的查询方式. public IList< ...

  5. 01-07-01【Nhibernate (版本3.3.1.4000) 出入江湖】并发控制

    Nhibernate 并发控制 [1]悲观并发控制 正在使用数据的操作,加上锁,使用完后解锁释放资源. 使用场景:数据竞争激烈,锁的成本低于回滚事务的成本 缺点:阻塞,可能死锁 [2]乐观并发控制: ...

  6. 01-08-03【Nhibernate (版本3.3.1.4000) 出入江湖】二级缓存:NHibernate自带的HashtableProvider之缓存管理

    http://www.cnblogs.com/lyj/archive/2008/11/28/1343418.html 管理NHibernate二级缓存 NHibernate二级缓存由ISessionF ...

  7. 01-08-04【Nhibernate (版本3.3.1.4000) 出入江湖】二级缓存:NHibernate自带的HashtableProvider之命名缓存

    http://www.cnblogs.com/lyj/archive/2008/11/28/1343418.html 可以在映射文件中定义命名查询,<query>元素提供了很多属性,可以用 ...

  8. 01-08-02【Nhibernate (版本3.3.1.4000) 出入江湖】二级缓存:NHibernate自带的HashtableProvider

    第一步骤:hibernate.cfg.xml文件补上如下配置: <?xml version="1.0" encoding="utf-8"?> < ...

  9. 01-08-01【Nhibernate (版本3.3.1.4000) 出入江湖】NHibernate中的三种状态

    以下属于不明来源资料: 引入 在程序运行过程中使用对象的方式对数据库进行操作,这必然会产生一系列的持久化类的实例对象.这些对象可能是刚刚创建并准备存储的,也可能是从数据库中查询的,为了区分这些对象,根 ...

随机推荐

  1. python基础:自定义函数

    一.背景 在学习函数之前,一直遵循:面向过程编程,即:根据业务逻辑从上到下实现功能,其往往用一长段代码来实现指定功能,开发过程中最常见的操作就是粘贴复制,也就是将之前实现的代码块复制到现需功能处,如下 ...

  2. (转)[转]大数据时代的 9 大Key-Value存储数据库

    在过去的十年中,计算世界已经改变.现在不仅在大公司,甚至一些小公司也积累了TB量级的数据.各种规模的组织开始有了处理大数据的需求,而目前关系型数据库在可缩放方面几乎已经达到极限. 一个解决方案是使用键 ...

  3. 冒泡,快排算法之javascript初体验

    引子:javascript实际使用的排序算法在标准中没有定义,可能是冒泡或快排.不用数组原生的 sort() 方法来实现冒泡和快排. Part 1:冒泡排序(Bubble Sort) 原理:临近的两数 ...

  4. 【原创】微信公众号与HTML 5混合模式揭秘3——JSSDK获取地理位置

    微信公众号与HTML 5混合模式揭秘1——如何部署JSSDK 微信公众号与HTML 5混合模式揭秘2——分享手机相册中照片 JSSDK为开发者提供了两个位置API服务:获取地理位置和使用微信内置地图查 ...

  5. 7款纯CSS3实现的炫酷动画应用

    1.纯CSS3实现人物摇头动画 这次我们要来分享一款超级可爱的纯CSS3人物摇头动画,初始化的时候人物的各个部位是利用CSS3动画效果拼接而成,接下来就是人物听音乐的场景,一边听音乐一边摇着脑袋,十分 ...

  6. 状态模式(State)

    状态模式,从字面意思上来讲应该是很简单的,就是针对实际业务上的内容,当类的内部的状态发生改变时,给出不同的响应体,就像现实中的人一样,早上没有吃饭,状态不好,上班.上课都会打哈欠,中午了,吃过午饭,又 ...

  7. Logstash+kibana+ ElasticSearch+redis

    这是之前Logstash+kibana+ ElasticSearch+redis 安装时,自己整理的初学者容易看懂的资料,按照以下的步骤也已经完成了安装. 这里有二台服务器: 192.168.148. ...

  8. 一款jQuery打造的滚动条在底部滑出信息提示层

    一款jQuery打造的滚动条在底部滑出信息提示层, 当滚动鼠标滚轮,或者滚动条往下拉的时候,在右下角,弹出一个信息提示框. 有一点仿的是一个插件工具,就是网页中大家都长用到的友荐. 这款特效算一款简单 ...

  9. Ping N个IP测试网络连通性

    #-----------------------Smokeping移动节点-------------------##! /bin/bashecho "------------- Statin ...

  10. 常用HTML meta 标签属性(网站兼容与优化需要),meta标签

    常用HTML meta 标签属性(网站兼容与优化需要),meta标签 标签提供关于HTML文档的元数据.元数据不会显示在页面上,但是对于机器是可读的.它可用于浏览器(如何显示内容或重新加载页面),搜索 ...