码迷,mamicode.com
首页 > 编程语言 > 详细

【java】itoo项目实战之hibernate 懒加载优化性能

时间:2015-07-26 09:49:38      阅读:138      评论:0      收藏:0      [点我收藏+]

标签:

      在做itoo 3.0 的时候,考评系统想要上线,就开始导入数据了,只导入学生2万条数据,但是导入的速度特别的慢,这个慢的原因是因为导入的时候进行了过多的IO操作。但是导入成功之后,查询学生的速度更加慢,因为底层用了hibernatehql语句进行查询的,学习过hibernate的人都知道,如果hibernate不设置懒加载的话,只有是有关联的数据都会一次性全部都查询出来,我试了试,查询2万条数据,最深的级联查询是有5层,然后发出来的语句是460条,时间大概是10s。然后就考虑使用lazy进行优化。然后就开始对lazy进行了一个研究。

问:懒加载可以提高性能吗? 

      不可以简单的说"能",因为Hibernate的关系映射拖累了SQL的性能,所以想出懒加载来弥补.只是弥补而以,不会超越.所以大家不要想着使用了懒加载总体性能就提高了,其实总体性能不下降就万幸了.

 

问:Hibernate的lazy属性可以配置在哪里?

(常用)<set><list>标签上 ,可以取值true,false,extra默认为true,当为true时,会懒加载,访问集合属性时再发出SQL语句.但set.size()时,很不智能,不会生成count()语句,而是查出所有记录赋值给set.

extra说明:调用集合的set.size()时,会生成selectcount(*)from tableName,比较智能.建议使用

(少用)<many-to-one><one-to-one>标签上 ,可以取值false,proxy,noproxy 当为true时,会有懒加载特性,当为false时会产生N+1问题,比如一个学生对应一个班级,用一条SQL查出10个学生,当访问学生的班级属性时Hibernate会再产生10条SQL分别查出每个学生对应的班级.

(不用)<class>标签上 ,可以取值true,false 默认为true,当为false时,load()方法将失去懒加载的特性与get()一样,不影响集合(<set><list>)标签上的lazy特性

(不用)<property>标签上 ,可以取值true,false  默认值为false,懒加载某个字段,无意义,不要使用

 

 

问:get()与load()的区别 

get()无懒加载特性,马上执行SQL查询.

load()有懒加载特性,会返加一个代理对象,所以永远不为null,先不执行SQL,要取对象的值时才执行SQL语句,前题session不能关闭,<class>标签上lazy不为false.

 

问:使用懒加载的条件

1 PO不能是final的

2能实现懒加载的对象(PO)都是被CGLIB改写的代理对象,所以不能是final修饰的

3须要asm,cglib两个jar包

4相应的lazy属性为true

5 相应的fetch属性为select 

 

问:什么时候遇到懒加载

1 使用load()

2 一对一<one-to-one> 

查主对象默认使用join连接,不会发生懒加载

查从对象默认会发生懒加载,先执行一句select查出从对象,当通过从对象访问了主对象时,再执行一句select查出主对象.

 

3多对一<many-to-one>

在使用hbm.xnl时,取多的一方时,默认会懒加载,不取一的一方

在使用JPA时,取多的一方时,默认自动使用joinon语句取出一的一方(用户与组,用户是多的一方,组是一的一方)

 

4一对多(<set><list>)

默认会懒加载,这是必须的,是重常用的。

 

问:实现懒加载的方案有哪些?

方法一:(没有使用懒加载)   

用 Hibernate.initialize(de.getEmps()) 提前加载一下. 

 

方法二:

把与Session脱离的对象重新绑定

lock()方法是用来让应用程序把一个未修改的对象重新关联到新session的方法。

//直接重新关联

ssion.lock(fritz,LockMode.NONE);

//进行版本检查后关联

session.lock(izi,LockMode.READ);

//使用SELECT ... FORUPDATE进行版本检查后关联

session.lock(pk,LockMode.UPGRADE);

 

方法三:

OpenSessionInView

OpenSessionInViewFilter是Spring提供的一个针对Hibernate的一个支持类,其主要意思是在发起一个页面请求时打开Hibernate的Session,一直保持这个Session,直到这个请求结束,具体是通过一个Filter来实现的。

 

由于Hibernate引入了LazyLoad特性,使得脱离Hibernate的Session周期的对象如果再想通过getter方法取到其关联对象的值,Hibernate会抛出一个LazyLoad的Exception。所以为了解决这个问题,Spring引入了这个Filter,使得Hibernate的Session的生命周期变长。

 

具体参考:http://www.iteye.com/topic/32001

 

问:fetch 和lazy 配置如何用于数据的查询

lazy 参数值常见有 false 和 true,Hibernate3 映射文件中默认lazy = true ; 

 

fetch 指定了关联对象抓取的方式,参数值常见是select和join,默认是select,select方式先查询主对象,再根据关联外键,每一个对象发一个select查询,获取关联的对象,形成了n+1次查询;而join方式,是left outerjoin查询,主对象和关联对象用一句外键关联的sql同时查询出来,不会形成多次查询。 

 

在映射文件中,不同的组合会使用不同的查询: 

1、lazy="true" fetch = "select",使用延迟策略,开始只查询出主对象,关联对象不会查询,只有当用到的时候才会发出sql语句去查询 ; 

 

2、lazy="false" fetch = "select",没有用延迟策略,同时查询出主对象和关联对象,产生1+n条sql. 

 

3、lazy="true"或lazy="false" fetch ="join",延迟都不会作用,因为采用的是外连接查询,同时把主对象和关联对象都查询出来了. 

 

      另外,在hql查询中,配置文件中设置的join方式是不起作用的,而在其他查询方式如get、criteria等是有效的,使用select方式;除非在hql中指定join fetch某个关联对象。fetch策略用于get/load一个对象时,如何获取非lazy的对象/集合。这些参数在Query中无效。

 

      最后因为前端框架使用的是easyui,绑定数据的时候,后台的数据需要转换成为json串,使用懒加载的话,级联查询数据在转换的时候就会出问题,所以UI就把懒加载的方式舍弃了,下一节未大家介绍另一种方式优化。

版权声明:本文为博主原创文章,未经博主允许不得转载。

【java】itoo项目实战之hibernate 懒加载优化性能

标签:

原文地址:http://blog.csdn.net/tang_huan_11/article/details/47065173

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!