概括潜在的Hibernate性能问题
创始人
2024-06-08 00:21:48
0

学习Hibernate时,经常会遇到Hibernate性能问题,这里将介绍Hibernate性能问题的解决方法。

在使用Hibernate进行分页的过程中,如果你收到如下警告,那么这里就是一个潜在的Hibernate性能问题点:

WARNING: firstResult/maxResults specified with collection fetch; applying in memory!

出现这个警告的直接后果是:无论你想要看第几页的数据,从Hibernate打印出的SQL来看它总是查询了所有满足条件的结果。这是为什么呢?来看看这句警告所在的代码,它位于org.hibernate.hql.ast.QueryTranslatorImpl中,部分摘录如下:

  1. view plaincopy to clipboardprint?  
  2. QueryNode query = ( QueryNode ) sqlAst;  
  3. boolean hasLimit = queryParameters.getRowSelection() != null && 
    queryParameters.getRowSelection().definesLimits();  
  4. boolean needsDistincting = ( query.getSelectClause().isDistinct() || hasLimit ) && 
    containsCollectionFetches();  
  5. QueryParameters queryParametersToUse;  
  6. if ( hasLimit && containsCollectionFetches() ) {  
  7. log.warn( "firstResult/maxResults specified with collection fetch; applying in memory!" )  
  8. RowSelection selection = new RowSelection();  
  9. selection.setFetchSize( queryParameters.getRowSelection().getFetchSize() );  
  10. selection.setTimeout( queryParameters.getRowSelection().getTimeout() );  
  11. queryParametersqueryParametersToUse = queryParameters.createCopyUsing( selection );  
  12. }  
  13. else {  
  14. queryParametersqueryParametersToUse = queryParameters;  
  15. }  
  16. List results = queryLoader.list( session, queryParametersToUse );  
  17. QueryNode query = ( QueryNode ) sqlAst;  
  18. boolean hasLimit = queryParameters.getRowSelection() != null && 
    queryParameters.getRowSelection().definesLimits();  
  19. boolean needsDistincting = ( query.getSelectClause().isDistinct() || hasLimit ) && 
    containsCollectionFetches();  
  20. QueryParameters queryParametersToUse;  
  21. if ( hasLimit && containsCollectionFetches() ) {  
  22. log.warn( "firstResult/maxResults specified with collection fetch; applying in memory!" );  
  23. RowSelection selection = new RowSelection();  
  24. selection.setFetchSize( queryParameters.getRowSelection().getFetchSize() );  
  25. selection.setTimeout( queryParameters.getRowSelection().getTimeout() );  
  26. queryParametersqueryParametersToUse = queryParameters.createCopyUsing( selection );  
  27. }  
  28. else {  
  29. queryParametersqueryParametersToUse = queryParameters;  
  30. }  
  31. List results = queryLoader.list( session, queryParametersToUse );  

关键在于if ( hasLimit && containsCollectionFetches() 这句判断,如果满足了这个条件,RowSelection将会被重新生成,原本分页需要的firstRow和maxRows属性将会丢失,后面的数据库分页自然也无法进行。Hibernate这么做的原因从代码上也很容易理解,如果查询需要限制条数(limit/offset)并且需要fetch结合对象,则重新生成RowSelection,进一步解释,就是当一个实体(A)和另一个实体(B)是One-To-Many关系的时候,一个需要fetch的典型查询语句是“select distinct a from A a left join fetch a.b”,由于1个A可能对应多个B,这个时候数据库查询的结果条数和需要生成的A对象的条数可能不一致,所以无法利用数据库层的分页来实现,因为你真正想分页的是A而不是A left join B。出现这个警告就是提醒你这个查询实际上是查询了所有满足条件的数据,Hibernate是在内存中对其进行了假分页的处理。

这样,对于查询结果比较多的情况无疑是一个Hibernate性能上的潜在威胁。碰到这样的情况,将Many的查询进行分开也是一种解决办法。

【编辑推荐】

  1. 分析Hibernate插入操作
  2. 剖析Hibernate批量更新
  3. 全面讲解Hibernate二级缓存
  4. 讲述Hibernate核心接口
  5. 概括Hibernate批量处理

相关内容

热门资讯

如何允许远程连接到MySQL数... [[277004]]【51CTO.com快译】默认情况下,MySQL服务器仅侦听来自localhos...
如何利用交换机和端口设置来管理... 在网络管理中,总是有些人让管理员头疼。下面我们就将介绍一下一个网管员利用交换机以及端口设置等来进行D...
施耐德电气数据中心整体解决方案... 近日,全球能效管理专家施耐德电气正式启动大型体验活动“能效中国行——2012卡车巡展”,作为该活动的...
20个非常棒的扁平设计免费资源 Apple设备的平面图标PSD免费平板UI 平板UI套件24平图标Freen平板UI套件PSD径向平...
德国电信门户网站可实时显示全球... 德国电信周三推出一个门户网站,直观地实时提供其安装在全球各地的传感器网络检测到的网络攻击状况。该网站...
为啥国人偏爱 Mybatis,... 关于 SQL 和 ORM 的争论,永远都不会终止,我也一直在思考这个问题。昨天又跟群里的小伙伴进行...
《非诚勿扰》红人闫凤娇被曝厕所... 【51CTO.com 综合消息360安全专家提醒说,“闫凤娇”、“非诚勿扰”已经被黑客盯上成为了“木...
2012年第四季度互联网状况报... [[71653]]  北京时间4月25日消息,据国外媒体报道,全球知名的云平台公司Akamai Te...