Hibernate学习笔记 第五章 检索策略

来源:互联网 发布:小米笔记本安装linux 编辑:程序博客网 时间:2024/06/05 18:24
【检索策略概述】
*当Hibernate从数据库中加载Customer对象时,如果同时加载所有关联的Order对象,而程序实际上仅仅需要访问Customer对象,那么这些


关联的Order对象就白白浪费了许多内存。
【类级别的检索策略】
*类级别可选的检索策略包括立即检索和延迟检索,默认为延迟检索
  - 立即检索:立刻加载检索方法指定的对象
  - 延迟检索:延迟加载检索方法指定的对象
*类级别的检索策略可以通过<class>元素的lazy属性进行设置
*如果程序加载一个对象的目的是为了访问它的属性,可以采取立即检索。如果程序加载一个持久化对象的目的是仅仅为了获得它的引用,


可以采用延迟检索
*无论<class>元素的<lazy>属性是true还是false,Session的get()放阿福及Query的list()放阿福在类级别总是使用立即检索策略
*若<class>元素的lazy属性为true或取默认值,Session的load()放阿福不会执行查询数据表的SELECT语句,仅返回代理类对象的实例,该代


理类是理由如下特征:
  - 有Hibernate在运行时采用CGLIB工具动态生成
  - Hibernate创建代理工具时,仅初始化其OID属性
  - 在应用程序第一次访问代理类市里的非OID属性时,Hibernate会初始化代理类实例
【一对多和多对多的检索策略】
*在映射文件中,用<set>元素来配置一对多关联及多对多关联关系。<set>元素有lazy和fetch属性
  - lazy:主要决定orders集合被初始化的时机。即到底是在加载Customer对象时就被初始化,还是在程序访问orders集合时被初始化
  - fetch:取值为"select"或"subselect"时,决定初始化orders的查询语句的形式;若取值为"join",则决定orders集合被初始化的时机
  - 若把fetch设置为"join",lazy属性将被忽略
【<set>元素的lazy和fetch属性】
|------------------------------------------------------------------------------------------------------||  lazy属性      |fetch属性       |                                                                    ||(默认值为true)  |(默认值为select)|                       检索策略                                     ||----------------|----------------|--------------------------------------------------------------------||     true       |   未显示设置   |采用延迟检索策略策略,这是默认的检索策略,也是有限考虑使用的检索策略||                |(取默认值select)|                                                                    ||----------------|----------------|--------------------------------------------------------------------||     false      |   未显示设置   |采用立即检索策略,当使用Hibernate二级缓存时,可以考虑使用立即检索   ||                |(取默认值select)|                                                                    ||----------------|----------------|--------------------------------------------------------------------||     extra      |   未显示设置   |采用加强延迟检索策略,它尽可能延迟orders集合被初始化的时机          ||                |(取默认值select)|                                                                    ||----------------|----------------|--------------------------------------------------------------------||  true,false    |   未显示设置   |lazy属性决定采用的检索策略,即决定初始化orders集合的时机,fetch属性为||  或extra       |(取默认值select)|select,意味着通过select语句来初始化orders集合形式为                 ||                |                |SELECT * FROM orders WHERE customer_id IN (1,2,3,4)            | |----------------|----------------|--------------------------------------------------------------------||  true,false    |    subselect   |lazy属性决定采用的检索策略,即决定初始化orders集合的时机.fetch属性为 ||  或extra       |                |subselect,意味着通过subselect语句来初始化orders集合,形式为         ||                |                |SELECT * FROM orders WHERE customer_id IN (SELECT id FROM customers)||----------------|----------------|--------------------------------------------------------------------||   未显示设置   |      join      |  采用迫切做外链接策略                                              ||(取默认值select)|                |                                                                    ||------------------------------------------------------------------------------------------------------|

【延迟检索和增强延迟检索】
*在延迟检索(lazy属性值为true)集合属性时,Hibernate在以下情况下初始化集合代理类实例
  - 应用程序第一次访问集合属性:iterator(),size(),isEmpty(),contains()等方法
  - 通过Hibernate.initialize()静态方法显示初始化
*增强延迟检索(lazy属性为extra):与lazy="true"类似。主要区别是增强延迟检索策略能进一步延迟Customer对象的orders集合代理市里的


初始化时机:
  - 当程序第一次访问orders属性的iterator()方法时,会导致orders集合代理类实例的初始化
  - 当程序第一次访问order属性的size(),contains()和isEmpty()方法时,Hibernate不会初始化orders集合类的实例,仅通过特定


的select语句查询必要的信息,不会检索所有的Order对象。
【<set>元素的batch-size属性】
*<set>元素有一个batch-size属性,用来为延迟检索策略或立即检索策略设定批量检索的数量。批量检索能减少SELECT语句的数目,提高延


迟检索或立即检索的运行性能。
【用带子查询的select语句正批量初始化orders集合(fetch属性为"subselect")】
*<set>元素的fetch属性:取值为"select"或"subselect"时,决定初始化orders的查询语句的形式;若取值为"join",则决定orders集合被


初始化的时机。默认值为select
*当fetch属性为"subselect"时
  - 假定Session缓存中有n个orders集合代理类实例没有被初始化,Hibernate能够通过带子查询的select语句,来批量初始化n个orders集


合代理类实例
  - batch-size属性将被忽略
  - 子查询中的select语句为最初查询CUSTOMERS表的SELECT语句
【迫切左外连接检索(fetch属性值设为"join")】
*<set>元素的fetch属性:取值为"select"或"subselect"时,决定初始化orders的查询语句的形式;若取值为"join",则决定orders集合被初始化的时机。默认值为select
*当fetch属性为"join"时:
  - 检索Customer对象时,会采用破鞋左外连接(通过左外连接加载与检索指定的对象关联的对象)策略来检索所有关联的Order对象
  - lazy属性将被忽略
  - Query的list()方法会忽略映射文件中配置的迫切左外连接检索策略,而依旧采用延迟加载策略
【多对一和一对一关联的检索策略】
*和<set>一样,<many-to-one>元素也有一个lazy属性和fetch属性。
|-----------------------------------------------------------||  lazy属性     |  fetch属性     |  检索Order对象时对关联的 ||(默认值为proxy)|(默认值为select)|Customer对象使用的检索策略||---------------|----------------|--------------------------||    proxy      |   未显示设置   |       采用延迟检索       ||               |(取默认值select)|                          ||---------------|----------------|--------------------------||   no-proxy    |   未显示设置   |      无代理延迟检索      ||               |(取默认值select)|                          ||---------------|----------------|--------------------------||    FALSE      |   未显示设置   |          立即检索        ||               |(取默认值select)|                          ||---------------|----------------|--------------------------||   未显示设置  |       join     |     迫切左外连接策略     ||(取默认值proxy)|                |                          ||-----------------------------------------------------------|

  - 若fetch属性设为join,那么lazy属性被忽略
  - 迫切左外连接检索策略的优点在于比立即检索策略使用的SELECT语句更少。
  - 无代理眼之间所需要增强持久化类的字节码才能实现
*Query的list方法会忽略映射文件配置的迫切左外连接检索策略,而采用延迟检索策略
*如果在关联级别使用了延迟加载或立即加载检索策略,可以设定批量检索的大小,以帮助提高延迟检索或立即检索的运行性能。
*Hibernate允许在应用程序中覆盖映射文件中设定的检索策略。
0 0