一、Hibernate的抓取策略
Hibernate抓取策略是指当应用程序需要在关联关系间进行导航的时候,Hibernate如何获取关联对象的策略。
Hibernate有如下几种抓取策略:
1、链接抓取(Join fetching):Hibernate 通过在select语句中使用out join来获取对象的关联实例或者关联集合。
2、查询抓取(Select fetching):发送另外一条select语句抓取当前对象的关联实体或者关联集合。除非我们显示地指定lazy=”false”禁止延迟抓取,否则只有当我们真正访问了关联关系的时候才会执行第二条select语句。
3、子查询抓取:另外发送一条SELECT语句抓取在前面查询到(或者抓取到)的所有实体对象的关联集合。除非你显式的指定lazy="false"禁止延迟抓取(lazyfetching),否则只有当你真正访问关联关系的时候,才会执行第二条select语句。
4、批量抓取(Batch fetching)- 对查询抓取的优化方案, 通过指定一个主键或外键列表,Hibernate使用单条SELECT语句获取一批对象实例或集合。
Hibernate会区分下列各种情况:
1、Immediate fetching,立即抓取- 当宿主被加载时,关联、集合或属性被立即抓取。
2、Lazy collectionfetching,延迟集合抓取- 直到应用程序对集合进行了一次操作时,集合才被抓取。(对集合而言这是默认行为。)
3、"Extra-lazy"collection fetching,"Extra-lazy"集合抓取-对集合类中的每个元素而言,都是直到需要时才去访问数据库。除非绝对必要,Hibernate不会试图去把整个集合都抓取到内存里来(适用于非常大的集合)。
4、Proxy fetching,代理抓取- 对返回单值的关联而言,当其某个方法被调用,而非对其关键字进行get操作时才抓取。
5、"No-proxy"fetching,非代理抓取- 对返回单值的关联而言,当实例变量被访问的时候进行抓取。与上面的代理抓取相比,这种方法没有那么“延迟”得厉害(就算只访问标识符,也会导致关联抓取)但是更加透明,因为对应用程序来说,不再看到proxy。这种方法需要在编译期间进行字节码增强操作,因此很少需要用到。
6、Lazy attributefetching,属性延迟加载- 对属性或返回单值的关联而言,当其实例变量被访问的时候进行抓取。需要编译期字节码强化,因此这一方法很少是必要的。
二、选择合理的抓取策略
定制合理的抓取策略对系统的提升有很大的作用。
查询抓取在N+1查询的情况下是极其脆弱的,因此我们可能会要求在映射文件中定义连接抓取(fetch=”join”)。
但是在映射文件中定义的抓取策略将会产生以下影响:通过get()或者load()方法获取数据。。只有在关联之间进行导航时,才会隐式的取得数据。
条件查询,使用了subselect抓取的HQL查询
不管使用哪种抓取策略,定义为非延时的类图会保证一定装载入内存,这就意味着一条HQL查询后紧跟着一系列的查询。
所以我们一般是这样做:通常情况下,我们并不使用映射文件进行抓取策略的定制,更多的是,保持其默认值然后在待定事物中,适用HQL的左连接对其进行重载。
Hibernate推荐的做法也是最佳实践:把所有对象关联的抓取都设为lazy!然后在特定事务中进行重载。
这种考虑是基于:对象之间的关联式错综复杂的,有时候哪怕我们只是一个简单的load,也会导致很多对象被load出来!所以在Hibernate中,所有对象关联都是lazy的。
四、Hibernate的批量抓取
在Hibernate中,对于关联抓取,可以定义每次抓取数据的数量。批量地将数据载入内存,减少与数据库交互的次数。在应用程序中可以定义默认的关联抓取数量。
Hibernate提供了两种批量抓取方案:类级别和集合级别。
类级别的批量查询。如果一个Session中需要载入30个student实例,在student中拥有一个成员变量class,该class执行Class对象。如果lazy=“true”,如果们需要变量整个student集合,每一个student都需要getClass(),hibernate在默认情况下回执行30次select查询语句,得到Class对象。这个时候可以通过在映射文件的Class属性,batch-size。
<class name=”Class” batch-size=”15”>….</class>
这样Hibernate将只需要执行两次查询即可:15,15。
集合级别的批量查询。如果我们需要遍历Class所拥有的所有Student对象,在Session中需要载入30个Class对象,遍历Class集合将会引起30次Select查询,每次查询都会调用getStudents()。如果在Class的映射定义中,允许对Students进行批量抓取,则Hibernate就会预先加载整个集合。
<set name=”students” batch-size=”15”>…</set>