Hibernate進(jìn)行時

          有關(guān)Hibenrate及其相關(guān)工具的主頁
          隨筆 - 0, 文章 - 16, 評論 - 29, 引用 - 0
          數(shù)據(jù)加載中……

          Hibernate3.2對sqlserver2005查詢分頁的處理

                   對Hibernate的查詢分頁,想必大家都比較熟悉了。setFirstResult()和setMaxResults()就可以搞定。但是使用sqlserver的朋友發(fā)現(xiàn)了嗎,hibernate發(fā)送的分頁語句中總是會有令人心煩的"select top 數(shù)字" 這樣的字符串。比如你一頁顯示50條記錄,現(xiàn)在要查詢第100頁的數(shù)據(jù),則會出現(xiàn)"select top 50000"這樣的語句,它是先把前5000條數(shù)據(jù)抓出到內(nèi)存中,處理后僅返回最后的50條給你,但是其他的4500條不是多余的了嗎?想想還真是憋火。
                   網(wǎng)上廣為流傳的一篇文章《實現(xiàn)Hibernate分頁查詢原理解讀》(作者robbin)中已經(jīng)講的很清楚了,如果數(shù)據(jù)庫自身支持分頁查詢,那么這種數(shù)據(jù)庫的Dialect中的supportsLimit()方法將返回true,而Hibernate則才會去調(diào)用getLimitString()方法以得到分頁的語句,比如對mysql來說是
          pagingSelect.append(" limit ?, ?");
          而對于oracle是:
          pagingSelect.append("select * from ( select row_.*, rownum rownum_ from ( ");
          當(dāng)然,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進(jìn)行分頁查詢的效率非常之低,遠(yuǎn)不如下面的語句:
          rs.absolute(firstRow);
          從《實現(xiàn)Hibernate分頁查詢原理解讀》中可以知道,rs.absolute(firstRow);執(zhí)行的條件是數(shù)據(jù)庫Dialect不支持分頁查詢,這句話有點繞,其實它的真正意思是“supportsLimit()方法返回的是false”。當(dāng)supportsLimit()返回false時,Hibernate采用rs.absolute(firstRow);來進(jìn)行分頁查詢。說到這里,大家心知肚明了吧。其實解決起來比較簡單,你自己定義一個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();
                      }

                  }

              }
          它和《實現(xiàn)Hibernate分頁查詢原理解讀》中描述的一樣,如果你的jdbc支持scrollable,那就調(diào)用rs.absolute(firstRow)定位到第一行,否則的話,就一行一行去移動吧。因此在配置文件中有一項很重要:
          <prop key="hibernate.jdbc.use_scrollable_resultset">true</prop>
          (注意我用的是spring的配置文件)上述這一行其實可以不寫,因為默認(rèn)就是true了,但如果你顯示地把它寫上了,一定要設(shè)為true,如果為false的話,則記錄集rs會一行一行去移動,還是很費事的。

          posted on 2007-08-06 15:15 caixuetao 閱讀(3688) 評論(4)  編輯  收藏

          評論

          # re: Hibernate3.2對sqlserver2005查詢分頁的處理  回復(fù)  更多評論   

          這幾天正在看蔡老師的Hibernate那本書啊!覺得寫得很好!
          這篇文章我轉(zhuǎn)載到我blog上拉,不會介意吧!
          2007-08-08 20:48 | 咖啡迷

          # re: Hibernate3.2對sqlserver2005查詢分頁的處理  回復(fù)  更多評論   

          我試了下好像不行, 全部都取了
          2008-06-26 11:30 | agua

          # re: Hibernate3.2對sqlserver2005查詢分頁的處理[未登錄]  回復(fù)  更多評論   

          用最少1k~10w的數(shù)據(jù)量測試,結(jié)果說明,通過調(diào)用rs.absolute(firstRow)實現(xiàn)分頁,對于性能的提高并沒有幫助。
          2010-03-08 17:55 | C

          # re: Hibernate3.2對sqlserver2005查詢分頁的處理  回復(fù)  更多評論   

          我的也是不起作用!!!!!!! 任何方言配置了 都不起作用!
          2011-05-09 09:49 | 劉玉海

          只有注冊用戶登錄后才能發(fā)表評論。


          網(wǎng)站導(dǎo)航:
           
          主站蜘蛛池模板: 若尔盖县| 黔东| 剑阁县| 基隆市| 金塔县| 襄汾县| 德令哈市| 海阳市| 西和县| 铁力市| 钦州市| 连江县| 兖州市| 沾益县| 水富县| 潼关县| 深州市| 襄樊市| 北碚区| 益阳市| 黔江区| 谷城县| 乌拉特前旗| 四平市| 大宁县| 元氏县| 井研县| 九江市| 乌拉特后旗| 丹江口市| 邯郸县| 班戈县| 吴川市| 故城县| 天台县| 寿光市| 山丹县| 海兴县| 穆棱市| 靖江市| 鄱阳县|