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

在做itoo 3.0
的时候,考评系统想要上线,就開始导入数据了,仅仅导入学生2万条数据,可是导入的速度特别的慢。这个慢的原因是由于导入的时候进行了过多的IO操作。可是导入成功之后,查询学生的速度更加慢。由于底层用了hibernate的hql语句进行查询的,学习过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后就把懒载入的方式舍弃了,下一节未大家介绍还有一种方式优化。

时间: 2024-10-10 19:51:18

【java】itoo项目实战之hibernate 懒载入优化性能的相关文章

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

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

【java】itoo项目实战之hibernate 批量保存优化

在itoo中.基本上每一个系统都有一个导入功能,大量的数据填写进入excel模板中.然后使用导入功能导入的数据库中,这样能够大大的提高工作效率. 那么导入就涉及到了批量保存数据库的问题了. 那么通常情况下,在一个Session对象的缓存中数量有限的持久化对象,等到Session对象处理事务完成,还要关闭Session对象,从而及时释放session的缓存占用的内存.在批量保存1万条数据,假设一次性把须要保存的1万条数据载入到内存职工,当运行事务提交的时候,就会清理缓存,hibernate运行1万

【java】itoo项目实战之常被忽视的性能优化

Itoo V3.0很快就要结束了,功能上基本上开发完成了,但是放到jboss中部署的时候,使用时感觉特别的慢,如果是数据量多的话,就把慢这个词发挥到了极致.这个慢的问题有大部分是因为基础系统中使用了JPA级联导致的,每次查询的时候,只要有关联的表,都会全部查询出来,一下发出一大版的HQL 语句,看着也是挺吓人的.出来优化JPA级联问题,还可以从代码中下手,从以下的几个方面考虑. (1)减少对象生命周期 对象的生命周期有这么一个计算公式:对象生命周期=销毁时间-创建时间 实际上减少对象生命周期有2

【java】itoo项目实战之大数据查询之使用 new map 优化hibernate之级联查询

在我的上一篇博客<[java]itoo项目实战之hibernate 懒加载优化性能>中,我曾提到过学生数据有2万条,查询数据十分的慢,这是让人很受不了的事情,看着页面进度条一直转着圈圈,那种着急的感觉真的没法形容.最开始考虑着使用lazy 来优化,因为前台框架的原因,lazy 优化并没有起到什么左右,后来就想着有select new map 优化.我先来画画关于查询学生的级联树 这个树的意思就是查询学生的时候它的深度是4级. 在没有优化之前,使用的是hibernate的hql 语句:From

【java】itoo项目实战之优化后具体代码

在我的前一篇博客中<<itoo项目实战之减少IO读写的导入思路>>,我介绍了如何完成减少IO读写的Excel导入,在这里我就把具体的代码实现分享给大家: 我就按照这张图的顺序给大家分享. 检查Excel 数据是否重复的代码: <span style="font-family:Times New Roman;font-size:18px;">// 2.从指定列中寻找重复行 for (int i = 1; i < realRows - 1; i++

Java Drp项目实战——Drp知多少

是什么 Drp是Distribution Resource Planning的缩写,意思是分销资源计划,它是用来管理企业的运行于Internet上的分销网络的系统,是以商业流程优化为基础,它的核心是销售和库存总和控制.这个分销系统或者说是分销体系,它的使用者包括一个大型企业的内部.各个分公司.各级分销商等,它的作用就在于即时的掌握各地的销售信息流.财务资金流.库存信息等一些功能. 产生背景是什么 知道了Drp是什么,我们还需要了解下它的开发背景是什么,为什么要开发这样的一个系统呢. 这个原因还是

JAVA Drp项目实战—— Unable to compile class for JSP 一波三折

交代下背景,电脑系统是64位的,用的是64位的Tomcat,安装是32位的Myeclipse10,java环境也是32位的,Tomcat在开始启动时会报这样一个错误,"Can't load IA 64-bit .dll on a AMD32-bit platform",但是不耽误使用,最近在敲Drp项目中用到了底层接口的几个方法,这个错误导致项目不能正常运行了,所以就将64位的Tomcat换成了与java环境一样的32位的Tomcat,上面的问题就顺利解决了,于是继续自己的开发,但是当

【java】itoo项目实战之百万数据查询优化收集与实践

1.对查询进行优化,应考虑在where 及 order by 涉及的列上建立索引. 2.应尽量避免在where 子句中对字段进行 null值判断,如:        select id from t wherenum is null 可以在num上设置默认值0,确保表中num列没有null值,然后这样查询: select idfrom t where num=0 3.应尽量避免在where 子句中使用!=或<>操作符. 4.应尽量避免在where 子句中使用 or 来连接条件,如: selec

【java】itoo项目实战之java Web 中web的作用

在ITOO中,每个war包下的web.xml都是一样的.它的结构如下: itoo中含有多个war包,而且每个war包中web.xml都和上图的结构一样,而且配置内容都是一样的. 在Javaweb 应用中可以包含HTML 文档,Servlet,JSP和相关java类等,为了Servlet容器能够顺利地找到javaWeb 应用中的各个组件,Servlet规范规定,javaWeb应用必须采取固定的目录结构,即每种类型的组件在Web应用中都有固定的存放目录.Servlet规范还规定,javaWeb应用的