延迟加载
概念
什么是延迟加载:所谓延迟加载就是当在真正需要数据的时候,才真正执行数据加载操作。可以简单理解为,只有在使用的时候,才会发出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类;三、使用带fetch的HQL语句
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"