01-05-01-2【Nhibernate (版本3.3.1.4000) 出入江湖】立即加载实现--NHibernateUtil.Initialize()和添加fetch关键字的HQL查询
相关资料:
http://www.cnblogs.com/lyj/archive/2008/10/29/1322373.html
问题的提出:
1.延迟加载,可能会引起session已经关闭的异常,例如:
当实现延迟加载,当需要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"}
解决方法是,使用强制立即加载:
方法一:为类添加一个方法专门用于加载Orders
public Orders LoadOrders(int customerId)
{
using (ISession _session = new SessionManager().GetSession())
{
return _session.Get<Customer>(customerId).Orders;
}
}
方法二:用NHibernateUtil.Initialize()进行强立即迟加载
public Customer Get(int customerId)
{
Customer customer = null;
ISession session = _sessionManager.GetSession();
ITransaction transaction = session.BeginTransaction();
try
{
customer = session.Get<Customer>(customerId);
//强制立即加载
NHibernateUtil.Initialize(customer.Orders);
transaction.Commit();
}
catch (Exception)
{
transaction.Rollback();
throw;
}
finally
{
session.Close();
} return customer;
} public Customer Load(int customerId)
{
Customer customer = null;
using (_session)
{
customer = _session.Load<Customer>(customerId);
//强制立即加载
NHibernateUtil.Initialize(customer.Orders);
return customer; }
}
问题的提出:
2.延迟加载,当关联的表很多,例如:
Customer--》Orders--》Product--》Company--》。。。
可能会引起连带加载过多其他实体对象,照成资源消耗。
解决方案:fetch的HQL查询:
【1】如果没有用关键fetch的HQL查询是对Orders进行延迟加载的,如下所示:
public IList<Customer> GetByHQL()
{
using (_session)
{
//注意;如果没有用关键fetch的HQL查询是对Orders进行延迟加载的,
return _session.CreateQuery("select DISTINCT c from Customer c join c.Orders ")
.List<Customer>(); }
} 测试代码:
[TestMethod]
public void TestGetByHQL()
{
CustomerService customerService = new CustomerService(); IList<Customer> customersList = customerService.GetByHQL(); //foreach (var customer in customersList)
//{
// Console.WriteLine("{0}的Order数量:{1}",
// customer.CustomerId, customer.Orders.Count());
//}
} 从下面输出的SQL语句,可以看出,没用加载Orders NHibernate:
select
distinct customer0_.CustomerId as CustomerId0_,
customer0_.Version as Version0_,
customer0_.Firstname as Firstname0_,
customer0_.Lastname as Lastname0_,
customer0_.Age as Age0_
from
Customer customer0_
inner join
[
Order] orders1_
on customer0_.CustomerId=orders1_.CustomerId
【2】在HQL查询中加入fetch进行强制立即加载
public IList<Customer> GetByHQLAndFetch()
{
using (_session)
{
//注意;如果没哟关键fetch,HQL是对Orders进行延迟加载的,
//就是通过fetch进行强制立即加载
return _session.CreateQuery("select DISTINCT c from Customer c join fetch c.Orders ")
.List<Customer>(); }
} 测试代码: [TestMethod]
public void TestGetByHQLAndFetch()
{
CustomerService customerService = new CustomerService(); IList<Customer> customersList = customerService.GetByHQLAndFetch(); //foreach (var customer in customersList)
//{
// Console.WriteLine("{0}的Order数量:{1}",
// customer.CustomerId, customer.Orders.Count());
//}
} 从下面输出的SQL语句,可以看出,添加了fetch的HQL对Orders进行了强制立即加载 NHibernate:
select
distinct customer0_.CustomerId as CustomerId0_0_,
orders1_.OrderId as OrderId1_1_,
customer0_.Version as Version0_0_,
customer0_.Firstname as Firstname0_0_,
customer0_.Lastname as Lastname0_0_,
customer0_.Age as Age0_0_,
orders1_.OrderDate as OrderDate1_1_,
orders1_.CustomerId as CustomerId1_1_,
orders1_.CustomerId as CustomerId0__,
orders1_.OrderId as OrderId0__
from
Customer customer0_
inner join
[
Order] orders1_
on customer0_.CustomerId=orders1_.CustomerId
fetch的缺陷,来自
http://www.cnblogs.com/lyj/archive/2008/10/29/1322373.html
使用HQL查询方法也可以立即加载。HQL语句支持的连接类型为:inner join(内连接)、left outer join(左外连接)、right outer join(右外连接)、full join(全连接,不常用)。
“抓取fetch”连接允许仅仅使用一个选择语句就将相关联的对象随着他们的父对象的初始化而被初始化,可以有效的代替了映射文件中的外联接与延迟属性声明。
几点注意:
- fetch不与setMaxResults() 或setFirstResult()共用,因为这些操作是基于结果集的,而在预先抓取集合时可能包含重复的数据,也就是说无法预先知道精确的行数。
- fetch还不能与独立的with条件一起使用。通过在一次查询中fetch多个集合,可以制造出笛卡尔积,因此请多加注意。对多对多映射来说,同时join fetch多个集合角色可能在某些情况下给出并非预期的结果,也请小心。
- 使用full join fetch 与 right join fetch是没有意义的。 如果你使用属性级别的延迟获取,在第一个查询中可以使用 fetch all properties 来强制NHibernate立即取得那些原本需要延迟加载的属性。
01-05-01-2【Nhibernate (版本3.3.1.4000) 出入江湖】立即加载实现--NHibernateUtil.Initialize()和添加fetch关键字的HQL查询的更多相关文章
- 01-08-05【Nhibernate (版本3.3.1.4000) 出入江湖】NHibernate二级缓存:第三方MemCache缓存
一.准备工作 [1]根据操作系统(位数)选择下载相应版本的MemCache, MemCache的下载和安装,参看: http://www.cnblogs.com/easy5weikai/p/37606 ...
- 01-03-02-2【Nhibernate (版本3.3.1.4000) 出入江湖】CRUP操作-Save方法的一些问题
此文由于当时不知道NHibernate的Sava方法不是更新操作,不知道Save就是Add,造成如下荒唐的求证过程,但结论是对的 ,可报废此文,特此声明. NHibernate--Save方法: Cu ...
- 01-08-01【Nhibernate (版本3.3.1.4000) 出入江湖】NHibernate中的一级缓存
缓存的范围? 1.事务范围 事务范围的缓存只能被当前事务访问,每个事务都有各自的缓存,缓存内的数据通常采用相互关联的对象形式.缓存的生命周期依赖于事务的生命周期,只有当事务结束时,缓存的生命周期才会结 ...
- 01-08-01【Nhibernate (版本3.3.1.4000) 出入江湖】NHibernate中的三种状态
以下属于不明来源资料: 引入 在程序运行过程中使用对象的方式对数据库进行操作,这必然会产生一系列的持久化类的实例对象.这些对象可能是刚刚创建并准备存储的,也可能是从数据库中查询的,为了区分这些对象,根 ...
- 01-05-01-1【Nhibernate (版本3.3.1.4000) 出入江湖】延迟加载及其class和集合(set、bag等)的Lazy属性配置组合对Get和Load方法的影响
这篇文章 http://ayende.com/blog/3988/nhibernate-the-difference-between-get-load-and-querying-by-id One o ...
- 01-04-03【Nhibernate (版本3.3.1.4000) 出入江湖】Criteria API关联查询
Criteria API关联查询 如果说HQL查询还有需要了解点SQL语法知识,并不是完全彻底面向对象查询, 那么Criterial API就是完全面向对象的查询方式. public IList< ...
- 01-04-02【Nhibernate (版本3.3.1.4000) 出入江湖】HQL查询
public IList<Customer> GetAllHql() { IList<Customer> result = null; ISession session = _ ...
- 01-07-01【Nhibernate (版本3.3.1.4000) 出入江湖】并发控制
Nhibernate 并发控制 [1]悲观并发控制 正在使用数据的操作,加上锁,使用完后解锁释放资源. 使用场景:数据竞争激烈,锁的成本低于回滚事务的成本 缺点:阻塞,可能死锁 [2]乐观并发控制: ...
- 01-08-03【Nhibernate (版本3.3.1.4000) 出入江湖】二级缓存:NHibernate自带的HashtableProvider之缓存管理
http://www.cnblogs.com/lyj/archive/2008/11/28/1343418.html 管理NHibernate二级缓存 NHibernate二级缓存由ISessionF ...
随机推荐
- UI1_UINavigationController
// // FourthViewController.h // UI1_UINavigationController // // Created by zhangxueming on 15/7/6. ...
- 位图文件格式及linux下C语言来操作位图文件
说到图片,位图(Bitmap)当然是最简单的,它是Windows显示图片的基本格式,其文件扩展名为*.BMP.由于没有经过任何的压缩,故BMP图片往往很大.在Windows下,任何格式的图片文件都要转 ...
- 10款很好用的 jQuery 图片滚动插件
jQuery 作为最流行的 JavaScript 框架,使用简单灵活,同时还有许多优秀的插件可供使用.其中最令人印象深刻的应用之一就是各种很酷的图片效果,它可以让的网站更具吸引力.这里收集了10款很好 ...
- 《JavaScript高级程序设计》心得笔记-----第二篇章
第五章 9.Function函数 1) 函数内部有两个特殊的对象: (1) arguments(主要用于保存函数参数,有一个属性callee,这是一个指针,指向拥有argu ...
- vs2010 配置OpenGL
为了之后的项目学习,需要学习OpenGL.在此进行下记录,方便查询. 准备工作: 到https://www.opengl.org/resources/libraries/glut/glutdlls37 ...
- Linux 静态库和动态库 使用说明
Linux下程序运行中,有两种库,静态库和动态库. 静态库:名字一般为libxxx.a,编译时会整合到可执行程序中,优点是运行时不需要外部函数库支持,缺点是编译后程序较大,一旦静态库改 ...
- 推荐一些C#相关的网站、资源和书籍
一.网站 1.http://msdn.microsoft.com/zh-CN/ 微软的官方网站,C#程序员必去的地方.那里有API开发文档,还有各种代码.资源下载. 2.http://social.m ...
- 初步了解SequoiaDB数据库
随着企业中日益复杂与多变的需求,以及迅速扩展带来的海量数据的业务,IT部门需要将越来越多的信息提供给用户,同时在现今的全球经济背景环境下,IT部 门还需要在提供高效服务的同时,降低其设备与程序维护成本 ...
- 简单的下拉刷新以及优化--SwipeRefreshLayout
代码工程简要说明:以一个SwipeRefreshLayout包裹ListView,SwipeRefreshLayout接管ListView的下拉事件,若ListView被用户触发下拉动作后,Swipe ...
- python之递归
递归的定义:即对自己自身内容的引用. 有用的递归函数应包含以下几步份: 当函数直接返回值时有基本的实例(最小可能性问题): 递归实例,包括一个或者多个问题较小部分的递归调用: 递归的关键就是将问题分解 ...