在传统的JDBC操作里,通常通过sql语句查询加载所需要的数据,当sql提交之后,这些数据也就被读取待用了,而在hibernate里,我们拥有了更多的数据加载的方式。以实现不同种的需求。
hibernate支持以下四种数据加载方式:
1、及时加载(Immediate Loading)--实体加载后,立即加载其关联数据。
2、延迟加载(Lazy Loading)--实体加载后,关联数据第一次被需要时,即访问然后加载。
3、预先加载(Eager Loading)--预先加载时,实体及其关联同时读取,于及时加载类似,但是通过一条sql(基于外连接查询)
4、批量加载(Batch Loading)--相对于及时和延迟加载,采用批量方式,可以进行性能上的优化。
一、及时加载
还是引入上篇中的User表和Address表,即一个用户可有多个住址。其配置中部分关联关系如下:
<set
name="address"
table="Address"
inverse="true"
cascade="none"
sort="unsorted"
lazy="false"
>
<key column="user_id"/>
<one-to-many class="com.entity.Address"/>
</set>
此时,再去做一个查询,代码如下:
String hql="from User where name='Erica' ";
List list=session.createQuery(hql).list();
syso("query finished");
Iterator it=list.iterator();
while(it.hasNext()){
User uesr=(User)it.next();
syso(user.getName());
syso(user.getAddresses().size());
}
在设置过show_sql后,控制台打印如下信息:
Hibernate:select......from User user where (name='Erica')
Hibernate:select......from Address addr where addr.user_id=?
query finished
Erica
2
从中我们可以得出,在执行find的时候,hibernate链接调用了2条sql语句,分别完成了对User和Address对象的加载。这就是及时加载的原理,当关联主体加载时(此处则我们的User对象),hibernate会立即自动读取其关联数据并完成关联属性的填充。
二、延时加载
延时加载就是对应上面及时加载所产生的,当我们不需要关联表的数据的时候,我们不需要查询,需要的时候,再发送语句,这样就避免了过多的性能的消耗。
将上面的部分配置的lazy属性改掉(hibernate3中,lazy属性默认为true,即默认启用延迟加载),即代码如下:
<set
name="address"
table="Address"
inverse="true"
cascade="none"
sort="unsorted"
lazy="true"
>
<key column="user_id"/>
<one-to-many class="com.entity.Address"/>
</set>
再次运行查询代码,我们发现控制台打印了如下的show_sql信息:
hibernate:select .....from User user where (name='Erica' )
query finished
Erica
hibernate:select ..... from Address addr where addr.user_id=?
2
与上面的不同的是,当代码运行到syso(user.getName());时,hibernate只有一条语句,而然在打印地址的个数的时候,激发了第二天sql语句去查询size()的大小。这就是他的特点,当真正需要关联表的数据的时候才会去做读取操作,从而提高性能。
三、预先加载
预先加载是通过outer join完成关联数据的加载,也是通过一条sql语句完成对实体以及其关联数据的读取操作,相对应即时加载的两条或多条,无疑提高了性能,但是一般只适用于一对一的关系。对于大部分的集合类型,还是推荐使用延迟加载的模式。一般而言,outer join可以提高处理的效率,但是对于关联关系复杂的,如多层关联,hibernate会生成非常复杂的sql语句,此时,我们应该根据实际情况,判断它的实用性,同时,也可以通过设置全局变量(hibernate.max_fetch_depth)限定outer-join的关联层次,一般5层比较合适。
四、批量加载
顾名思义,就是通过批量提交多个限定条件,一次完成多个数据读取,如对于下面的sql请求:select from User where id=1; select from User where id=2经过整合后变为,select from User where id=1 or id=2。如果使用了这种方式,hibernate在进行数据操作前,会自动在当前session中寻找,是否有其他同类型的待加载的数据,如果有,则将其合并在当前的select 语句中一起提交。
在实体的配置中,我们可以在class节点下,通过设置batch-size打开批量加载机制,并限定每次批量加载的数量,代码如下:
<class name="User" table="Uesr" batch-size="5">
分享到:
相关推荐
很多人都对Java在批量数据的处理方面是否是其合适的场所持有怀疑的念头,由此延伸,那么就会认为ORM可能也不是特别适合数据的批量处理。 其实,我想如果我们应用得当的话,完全可以消除ORM批量处理性能问题这方面的...
Hibernate延时加载与lazy机制.docHibernate延时加载与lazy机制.doc
NULL 博文链接:https://javakeith.iteye.com/blog/860140
在Hibernate中处理批量更新和批量删除
Hibernate延迟加载以及利用Spring 大家看看 参考一下
深入理解hibernate懒加载技术,正确使用懒加载
hibernate3的属性延时加载是个很有意义的东西,它能让你把你不想加载到内存里的东西在查询的时候排除。 我相信来下载这个东西的人都已经很了解这一块的知识了,只是配不对那个类增强器才来下载我这个文件。 这些...
详细介绍hibernate延迟加载,对hibernate初学者有一定的帮助
hibernate延迟加载解决 延迟加载的具体解决方法
使用Hibernate将大量记录插入到数据库
jsp中关于Hibernate延时加载的问题,实例源代码,详解都有
自己研究的hibernate加载方式和多态加载的分析. 写的不好,不过可以供深入者确切体会hibernate的种种功能的本质
通过延迟加载技术可以避免过多、过早地加载数据表里的数据,从而降低应用的内存开销。Hibernate 的延迟加载本质上就是代理模式的应用,当程序通过 Hibernate 装载一个实体时,默认情况下,Hibernate 并不会立即抓取...
hibernate懒加载策略.dochibernate懒加载策略.doc
在我们的Java项目中,批量更新是指在一个事务中更新大批量数据,批量删除是指在一个事务中删除大批量数据。批量删除虽然在Hibernate里也可以实现,但因Hibernate的实现机制是一个一个删除,在数量大的情况下很影响...
hibernate优化前的程序,Hibernate 延时加载,Hibernate数据加载方式
Hibernate延迟加载机制.zip
Hibernate延迟加载Hibernate延迟加载