<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    Hibernate進(jìn)行時(shí)

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

    Hibernate3.2對(duì)sqlserver2005查詢分頁(yè)的處理

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

            }

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

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

    評(píng)論

    # re: Hibernate3.2對(duì)sqlserver2005查詢分頁(yè)的處理  回復(fù)  更多評(píng)論   

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

    # re: Hibernate3.2對(duì)sqlserver2005查詢分頁(yè)的處理  回復(fù)  更多評(píng)論   

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

    # re: Hibernate3.2對(duì)sqlserver2005查詢分頁(yè)的處理[未登錄](méi)  回復(fù)  更多評(píng)論   

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

    # re: Hibernate3.2對(duì)sqlserver2005查詢分頁(yè)的處理  回復(fù)  更多評(píng)論   

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

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


    網(wǎng)站導(dǎo)航:
     
    主站蜘蛛池模板: 中文在线观看永久免费| 日韩亚洲国产高清免费视频| 国产精品亚洲综合天堂夜夜| 免费A级毛片无码免费视| 亚洲第一区视频在线观看| 久章草在线精品视频免费观看| 国产成人精品日本亚洲网站| 久久er国产精品免费观看2| 亚洲AV日韩精品久久久久 | 亚洲乱码在线卡一卡二卡新区| 67194国产精品免费观看| 亚洲欧洲视频在线观看| 成人免费黄色网址| 亚洲性线免费观看视频成熟| 日本媚薬痉挛在线观看免费| 久久亚洲色WWW成人欧美| 又黄又大又爽免费视频| 四虎精品成人免费视频| 亚洲伊人久久大香线蕉综合图片| 国产午夜成人免费看片无遮挡| 亚洲AV日韩AV天堂久久 | 久久精品国产亚洲AV| 亚洲精品专区在线观看| 91福利免费网站在线观看| 亚洲人成在线电影| 成年18网站免费视频网站| 老司机午夜性生免费福利| 中文字幕亚洲乱码熟女一区二区| 外国成人网在线观看免费视频| 亚洲一本之道高清乱码| 国产jizzjizz视频全部免费| 中文字幕的电影免费网站| 久久亚洲日韩看片无码| 天堂在线免费观看中文版| 一级毛片免费播放男男| 亚洲∧v久久久无码精品 | 成人AV免费网址在线观看| 一级做a爰黑人又硬又粗免费看51社区国产精品视 | 97精品免费视频| 国产精品无码亚洲精品2021| 久久久久亚洲精品无码系列|