Hibernate性能调优--集合属性的延迟加载

Hibernae的延迟加载是一个非常常用的技术,实体的集合属性默认会被延迟加载,实体所关联的实体默认也会被延迟加载。Hibernate通过这种延迟加载来降低系统的内存开销,从而保证 Hibernate 的运行性能。

下面先来剖析Hibernate 延迟加载的“秘密”。

当 Hibernate从数据库中初始化某个持久化实体时,该实体的集合属性是否随持久化类一起初始化呢?如果集合属性里包含十万,甚至百万的记录,在初始化持久化实体的同时,完成所有集合属性的抓取,将导致性能急剧下降。完全有可能系统只需要使用持久化类集合属性中的部分记录,而完全不是集合属性的全部,这样,没有必要一次加载所有的集合属性。

对于集合属性,通常推荐使用延迟加载策略。所谓延迟加载就是等系统需要使用集合属性时才从数据库装载关联的数据。

            public class Person{

       privateInteger id;

       privateString name;

       privateint age;

       privateSet<Address> addresses = new HashSet<Address>();

           }


为了让Hibernate 能管理该持久化类的集合属性,程序为该持久化类提供如下映射文件:

Person.hbm.xml

<hibernate-mappingpackage="org.crazyit.app.domain">

<!--映射Person持久化类 -->

<classname="Person" table="person_inf">

<idname="id" column="person_id">

<generatorclass="identity" />

</id>

<propertyname="name" type="string" />

<propertyname="age" type="int" />

<setname="addresses" table="person_address"lazy="true">

<keycolumn="person_id" />

<composite-elementclass="Address">

<propertyname="detail" />

<propertyname="zip" />

</composite-element>

</set>

</class>

</hibernate-mapping>

 

从上面映射文件的代码可以看出,Person的集合属性中的 Address 类只是一个普通的 POJO。该 Address 类里包含 detail、zip 两个属性。

上面映射文件中<set.../> 元素里的代码指定了 lazy="true"(对于 <set.../>元素来说,lazy="true"是默认值),它指定 Hibernate 会延迟加载集合属性里 Address 对象。

例如通过如下代码来加载ID 为 1 的 Person 实体:

<span style="font-size:18px;"> Session session = sessionFactory.getCurrentSession(); 
 Transaction tx =session.beginTransaction(); 
 Person p = (Person)session.get(Person.class, 1);  //<1>
 System.out.println(p.getName());</span>

上面代码只是需要访问ID 为 1 的 Person 实体,并不想访问这个 Person 实体所关联的 Address 对象。此时有两种情况:

  • 如果不延迟加载,Hibernate就会在加载 Person实体对应的数据记录时立即抓取它关联的 Address对象。
  • 如果采用延迟加载,Hibernate就只加载 Person实体对应的数据记录。

很明显,第二种做法既能减少与数据库的交互,而且避免了装载Address 实体带来的内存开销——这也是 Hibernate 默认启用延迟加载的原因。

现在的问题是,延迟加载到底是如何实现的呢?Hibernate 在加载 Person 实体时,Person 实体的 addresses 属性值是什么呢?

为了解决这个问题,我们在 <1>号代码处设置一个断点,在 Eclipse中进行 Debug,此时可以看到 Eclipse Console窗口有如图所示的输出:

延迟加载集合属性的 Console 输出

正如图输出所看到的,此时Hibernate 只从 Person 实体对应的数据表中抓取数据,并未从 Address 对象对应的数据表中抓取数据,这就是延迟加载。

那么Person 实体的 addresses 属性是什么呢?此时可以从 Eclipse 的 Variables 窗口看到如图所示的结果:

延迟加载的集合属性值

从图的方框里的内容可以看出,这个 addresses 属性并不是我们熟悉的 HashSet、TreeSet 等实现类,而是一个 PersistentSet实现类,这是 Hibernate 为 Set 接口提供的一个实现类。

PersistentSet集合对象并未真正抓取底层数据表的数据,因此自然也无法真正去初始化集合里的 Address 对象。不过 PersistentSet 集合里持有一个session 属性,这个 session 属性就是 Hibernate Session,当程序需要访问 PersistentSet集合元素时,PersistentSet 就会利用这个 session 属性去抓取实际的 Address 对象对应的数据记录。

那么到底抓取那些Address 实体对应的数据记录呢?这也难不倒 PersistentSet,因为 PersistentSet 集合里还有一个 owner属性,该属性就说明了 Address 对象所属的 Person 实体,Hibernate 就会去查找 Address 对应数据表中外键值参照到该 Person实体的数据。

例如我们单击图所示窗口中 addresses 行,也就是告诉 Eclipse 要调试、输出 addresses 属性,这就是要访问 addresses 属性了,此时就可以在Eclipse 的 Console 窗口看到输出如下 SQL 语句:

    select
        addresses0_.person_id asperson1_0_0_,
        addresses0_.detail as detail0_,
        addresses0_.zip as zip0_
    from
        person_address addresses0_
    where
        addresses0_.person_id=?

这就是PersistentSet 集合跟据 owner 属性去抓取特定 Address 记录的 SQL 语句。此时可以从 Eclipse 的 Variables窗口看到图所示的输出:

已加载的集合属性值

从图可以看出,此时的 addresses 属性已经被初始化了,集合里包含了 2 个 Address 对象,这正是 Person 实体所关联的两个 Address对象。

通过上面介绍可以看出,Hibernate对于 Set 属性延迟加载关键就在于 PersistentSet 实现类。在延迟加载时,开始 PersistentSet 集合里并不持有任何元素。但PersistentSet 会持有一个 Hibernate Session,它可以保证当程序需要访问该集合时“立即”去加载数据记录,并装入集合元素。

与PersistentSet 实现类类似的是,Hibernate 还提供了PersistentList、PersistentMap、PersistentSortedMap、PersistentSortedSet等实现类,它们的功能与 PersistentSet 的功能大致类似。

熟悉Hibernate 集合属性读者应该记得:Hibernate 要求声明集合属性只能用 Set、List、Map、SortedSet、SortedMap等接口,而不能用 HashSet、ArrayList、HashMap、TreeSet、TreeMap 等实现类,其原因就是因为 Hibernate需要对集合属性进行延迟加载,而 Hibernate 的延迟加载是依靠PersistentSet、PersistentList、PersistentMap、PersistentSortedMap、PersistentSortedSet来完成的——也就是说,Hibernate 底层需要使用自己的集合实现类来完成延迟加载,因此它要求开发者必须用集合接口、而不是集合实现类来声明集合属性。

Hibernate对集合属性默认采用延迟加载,在某些特殊的情况下,为 <set.../>、<list.../>、<map.../>等元素设置 lazy="false"属性来取消延迟加载。


评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

码海拾贝2023

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值