新书推介:《语义网技术体系》
作者:瞿裕忠,胡伟,程龚
   XML论坛     W3CHINA.ORG讨论区     计算机科学论坛     SOAChina论坛     Blog     开放翻译计划     新浪微博  
 
  • 首页
  • 登录
  • 注册
  • 软件下载
  • 资料下载
  • 核心成员
  • 帮助
  •   Add to Google

    >> 本版讨论Java, J2SE, J2ME, J2EE, 以及Eclipse, NetBeans, JBuilder等Java开发环境,还有JSP, JavaServlet, JavaBean, EJB以及struts, hibernate, spring, webwork2, Java 3D, JOGL等相关技术。
    [返回] 中文XML论坛 - 专业的XML技术讨论区计算机技术与应用『 Java/Eclipse 』 → 实现Hibernate分页查询原理解读 查看新帖用户列表

      发表一个新主题  发表一个新投票  回复主题  (订阅本版) 您是本帖的第 3307 个阅读者浏览上一篇主题  刷新本主题   树形显示贴子 浏览下一篇主题
     * 贴子主题: 实现Hibernate分页查询原理解读 举报  打印  推荐  IE收藏夹 
       本主题类别:     
     npuhetao 帅哥哟,离线,有人找我吗?
      
      
      威望:5
      头衔:骨灰级菜鸟
      等级:大二(研究C++)
      文章:230
      积分:2708
      门派:XML.ORG.CN
      注册:2004/3/18

    姓名:(无权查看)
    城市:(无权查看)
    院校:(无权查看)
    给npuhetao发送一个短消息 把npuhetao加入好友 查看npuhetao的个人资料 搜索npuhetao在『 Java/Eclipse 』的所有贴子 引用回复这个贴子 回复这个贴子 查看npuhetao的博客楼主
    发贴心情 实现Hibernate分页查询原理解读

    Hibernate 可以实现分页查询,例如:
    从第2万条开始取出100条记录

    Query q = session.createQuery("from Cat as c");
    q.setFirstResult(20000);
    q.setMaxResults(100);
    List l = q.list();

      那么Hibernate底层如何实现分页的呢?实际上Hibernate的查询定义在net.sf.hibernate.loader.Loader这个类里面,仔细阅读该类代码,就可以把问题彻底搞清楚。

      Hibernate2.0.3的Loader源代码第480行以下:

    if (useLimit) sql = dialect.getLimitString(sql);
    PreparedStatement st = session.getBatcher().prepareQueryStatement(sql, scrollable);

      如果相应的数据库定义了限定查询记录的sql语句,那么直接使用特定数据库的sql语句。

      然后来看net.sf.hibernate.dialect.MySQLDialect:

    public boolean supportsLimit() {
    return true;
    }
    public String getLimitString(String sql) {
    StringBuffer pagingSelect = new StringBuffer(100);
    pagingSelect.append(sql);
    pagingSelect.append(" limit ?, ?");
    return pagingSelect.toString();
    }

      这是MySQL的专用分页语句,再来看net.sf.hibernate.dialect.Oracle9Dialect:

    public boolean supportsLimit() {
    return true;
    }

    public String getLimitString(String sql) {
    StringBuffer pagingSelect = new StringBuffer(100);
    pagingSelect.append("select * from ( select row_.*, rownum rownum_ from ( ");
    pagingSelect.append(sql);
    pagingSelect.append(" ) row_ where rownum <= ?) where rownum_ > ?");
    return pagingSelect.toString();
    }

      Oracle采用嵌套3层的查询语句结合rownum来实现分页,这在Oracle上是最快的方式,如果只是一层或者两层的查询语句的rownum不能支持order by。

      除此之外,Interbase,PostgreSQL,HSQL也支持分页的sql语句,在相应的Dialect里面,大家自行参考。

      如果数据库不支持分页的SQL语句,那么根据在配置文件里面

    #hibernate.jdbc.use_scrollable_resultset true  

      默认是true,如果你不指定为false,那么Hibernate会使用JDBC2.0的scrollable result来实现分页,看Loader第430行以下:

    if ( session.getFactory().useScrollableResultSets() ) {
    // 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();
    }

      如果支持scrollable result,使用ResultSet的absolute方法直接移到查询起点,如果不支持的话,使用循环语句,rs.next一点点的移过去。

      可见使用Hibernate,在进行查询分页的操作上,是具有非常大的灵活性,Hibernate会首先尝试用特定数据库的分页sql,如果没用,再尝试Scrollable,如果不行,最后采用rset.next()移动的办法。

      在查询分页代码中使用Hibernate的一大好处是,既兼顾了查询分页的性能,同时又保证了代码在不同的数据库之间的可移植性。


       收藏   分享  
    顶(0)
      




    ----------------------------------------------
    你硬要把单纯的事情看得很严重,那样子你会很痛苦。

    点击查看用户来源及管理<br>发贴IP:*.*.*.* 2005/3/14 13:44:00
     
     GoogleAdSense
      
      
      等级:大一新生
      文章:1
      积分:50
      门派:无门无派
      院校:未填写
      注册:2007-01-01
    给Google AdSense发送一个短消息 把Google AdSense加入好友 查看Google AdSense的个人资料 搜索Google AdSense在『 Java/Eclipse 』的所有贴子 访问Google AdSense的主页 引用回复这个贴子 回复这个贴子 查看Google AdSense的博客广告
    2025/9/16 11:31:46

    本主题贴数1,分页: [1]

    管理选项修改tag | 锁定 | 解锁 | 提升 | 删除 | 移动 | 固顶 | 总固顶 | 奖励 | 惩罚 | 发布公告
    W3C Contributing Supporter! W 3 C h i n a ( since 2003 ) 旗 下 站 点
    苏ICP备05006046号《全国人大常委会关于维护互联网安全的决定》《计算机信息网络国际联网安全保护管理办法》
    46.875ms