`

Hibernate3.2对sqlserver2005查询分页的处理

阅读更多
对Hibernate的查询分页,想必大家都比较熟悉了。setFirstResult()和setMaxResults()就可以搞定。但是使用sqlserver的朋友发现了吗,hibernate发送的分页语句中总是会有令人心烦的"select top 数字" 这样的字符串。比如你一页显示50条记录,现在要查询第100页的数据,则会出现"select top 50000"这样的语句,它是先把前5000条数据抓出到内存中,处理后仅返回最后的50条给你,但是其他的4500条不是多余的了吗?想想还真是憋火。
          网上广为流传的一篇文章《实现Hibernate分页查询原理解读》(作者robbin)中已经讲的很清楚了,如果数据库自身支持分页查询,那么这种数据库的Dialect中的supportsLimit()方法将返回true,而Hibernate则才会去调用getLimitString()方法以得到分页的语句,比如对mysql来说是
pagingSelect.append(" limit ?, ?");而对于oracle是:
pagingSelect.append("select * from ( select row_.*, rownum rownum_ from ( ");当然,sqlserver也不甘落后,好像非得supportsLimit它才舒服。它的supportsLimit()也是true,同时它的getLimitString()方法为:
public String getLimitString(String querySelect, int offset, int limit) {
        if ( offset > 0 ) {
            throw new UnsupportedOperationException( "sql server has no offset" );
         }
        return new StringBuffer( querySelect.length()+8 )
             .append(querySelect)
             .insert( getAfterSelectInsertPoint(querySelect), " top " + limit )
             .toString();
     }看到那个“top”的来历了吧。
而实事上,select top进行分页查询的效率非常之低,远不如下面的语句:
rs.absolute(firstRow);从《实现Hibernate分页查询原理解读》中可以知道,rs.absolute(firstRow);执行的条件是数据库Dialect不支持分页查询,这句话有点绕,其实它的真正意思是“supportsLimit()方法返回的是false”。当supportsLimit()返回false时,Hibernate采用rs.absolute(firstRow);来进行分页查询。说到这里,大家心知肚明了吧。其实解决起来比较简单,你自己定义一个MySqlServer2005Dialect,继承于原来的org.hibernate.dialect.SQLServerDialect,覆盖其supportsLimit()方法,如下:
public boolean supportsLimit() {
        return false;
     }然后在hibernate配置文件中使用你自己的MySqlServer2005Dialect方言即可。
同时要注意的是,在Loader类的1471行有一个方法,它是:
/** *//**
      * Advance the cursor to the first required row of the <tt>ResultSet</tt>
     */
    private void advance(final ResultSet rs, final RowSelection selection)
            throws SQLException {

        final int firstRow = getFirstRow( selection );
        if ( firstRow != 0 ) {
            if ( getFactory().getSettings().isScrollableResultSetsEnabled() ) {
                // we can go straight to the first required row
                 rs.absolute( firstRow );
             }
            else {
                // we need to step through the rows one row at a time (slow)
                for ( int m = 0; m < firstRow; m++ ) rs.next();
             }
         }
     }它和《实现Hibernate分页查询原理解读》中描述的一样,如果你的jdbc支持scrollable,那就调用rs.absolute(firstRow)定位到第一行,否则的话,就一行一行去移动吧。因此在配置文件中有一项很重要:
<prop key="hibernate.jdbc.use_scrollable_resultset">true</prop>
(注意我用的是spring的配置文件)上述这一行其实可以不写,因为默认就是true了,但如果你显示地把它写上了,一定要设为true,如果为false的话,则记录集rs会一行一行去移动,还是很费事的。

分享到:
评论

相关推荐

    Struts2+Hibernate3.2+spring2.0+sqlserver2000

    Struts2、Hibernate3.2、Spring2.0和SQL Server 2000是Java Web开发中的四个关键组件,它们构成了一个强大的企业级应用框架,通常被称为SSH2(Struts2、Spring2、Hibernate2)集成框架。下面将详细阐述这些技术以及...

    Ext3.2的TreePanel和GridPanel的分页与Hibernate的分页功能的影射

    环境:Windows XP sp3、IE 7.0、MS SQL Server 2000、MyEclipse5.5、 Ext 3.2、Tomcat 6.0 使用步骤: 1、下载解压之后,使用IDE打开工程 2、在工程中找到“数据库脚本.sql”文档,然后在查询分析器中生成数据与测试...

    hibernate3.2_中文参考手册.pdf

    - **分页查询**:支持对查询结果进行分页处理,避免一次性加载大量数据。 #### 六、最佳实践与常见问题 - **合理设计实体类**:遵循单一职责原则,避免实体类过于臃肿。 - **正确使用缓存**:合理配置缓存策略,...

    Hibernate通用分页

    【标题】"Hibernate通用分页"涉及的是在Java开发中使用Hibernate框架进行数据查询时的分页处理技术。在Web应用程序中,为了提高用户体验,通常需要对大量的数据库记录进行分页展示,而不是一次性加载所有数据。这就...

    java分页技术汇总

    JSP分页的核心在于对数据源的切片处理,通常是在服务器端根据用户的请求参数(如当前页码、每页显示条数等)计算出具体的SQL查询语句或调用相应的数据库API来获取数据子集。 ##### 2.2 JSP分页示例 假设有一个`...

    java程序员简历模板

    * 标准 SQL 语句,Oracle,MySQL,SQLServer2005 数据库 * Ajax,jquery 等技术, html+css+javascript 和 xml * Flash,PS,Dreamweaver * JUnit,Debug 进行程序调试、log4j 进行日志管理、SVN 项目整合、MyBatis 数据库...

    struts1+spring+JPA+jstl的CRUD操作,包含分页

    在这个项目中,使用了Hibernate3.2作为JPA的实现,它是一个强大的ORM(对象关系映射)工具,能将Java类映射到数据库表,简化数据操作。JPA使得开发者可以使用面向对象的方式来处理数据库操作,而无需编写大量的SQL...

    NHibernate3.2 例子

    1. **支持多种数据库**:NHibernate 3.2支持多种数据库系统,如MySQL、Oracle、SQL Server、SQLite等。 2. **改进的查询语言(HQL)**:HQL是NHibernate的查询语言,类似于SQL,但面向对象,支持更复杂的查询。 3....

    基于J2EE框架的个人博客系统项目毕业设计论文(源码和论文)

    3.2.1. Web应用程序开发环境—SQLserver数据库 SQL Server是由Microsoft开发和推广的关系数据库管理系统(DBMS),它最初是由Microsoft、Sybase和Ashton-Tate三家公司共同开发的。SQL Server 2000是Microsoft公司于...

    《JSP程序开发范例宝典》完整目录程序清单

    - **实085**:介绍如何使用Hibernate框架连接SQL Server数据库。 **5.2 ACCESS数据库** - **实086**:演示如何通过JDBC-ODBC桥接方式连接ACCESS数据库。 - **实087**:介绍如何使用Hibernate框架连接ACCESS数据库...

    eSql语法学习资料

    ##### 11.1 使用SqlServer函数 调用SQL Server的内置函数。 ##### 11.2 使用.NET的数据类型 在查询中使用.NET框架提供的类型。 #### 十二、ESQL关系与导航 ESQL还支持关系和导航特性,允许在查询中方便地处理实体...

    java工程师简历.pdf

    - **技术栈**:Eclipse3.2+MyEclipse5.0+PowerDesigner+CVS+IReport+Windows2000+SQLServer2000+Weblogic8.1+JDK1.5+Struts+Spring+Hibernate - **项目描述**:该项目旨在帮助江西仁和药业提升销售效率、整合客户...

    医院管理系统是典型的管理信息系统(MIS),其开发主要包括后台数据库的建立和维护以及前端应用程序的开发两个方面.

    通过采用先进的开发工具和技术,如Hibernate+Struts结合SQL Server 2000,可以在较短时间内构建出初步的系统原型,并在此基础上不断地迭代优化,最终实现用户满意且功能完善的医院管理系统。 在整个开发过程中,...

    2017年尚学堂Java培训课程大纲.docx

    - **分页查询技术**:实现数据分页展示,提高用户体验。 - **EL表达式和JSTL标签库**:掌握Expression Language(EL)的使用方法,熟悉JavaServer Pages Standard Tag Library(JSTL)标签库。 - **Log4j日志组件的...

    基于ssm+jsp框架的中小企业人力资源管理系统源码数据库.doc

    - **输入过滤**:对用户提交的数据进行严格的格式校验,防止SQL注入等攻击。 - **数据加密**:敏感信息采用加密存储,如员工的身份证号、薪资等。 - **日志记录**:记录用户的操作日志,便于追踪异常行为。 #### 五...

    Java工程师面试宝典

    - **定义**:攻击者利用不当的输入验证,在SQL查询中插入恶意SQL语句。 - **防御措施**:使用参数化查询或预编译语句。 ##### 8.4 SQL Select语句完整的执行顺序 - **解析SQL语句**:语法分析。 - **优化器选择...

Global site tag (gtag.js) - Google Analytics