Sql Server 2012 分頁方法分析(offset and fetch)
更新時間:2012年08月09日 22:00:31 作者:
最近在分析 Sql Server 2012 中 offset and fetch 的新特性,發(fā)現 offset and fetch 無論語法的簡潔還是功能的強大,都是相當相當不錯的
其中 offset and fetch 最重要的新特性是 用來 分頁,既然要分析 分頁,就肯定要和之前的分頁方式來比較了,特別是 Row_Number() 了,在比較過程中,發(fā)現了蠻多,不過最重要的,通過比較本質,得出了優(yōu)劣,也和大家一起分享下。
準備工作,建立測試表:Article_Detail,主要是用來存放一些文章信息,測試的時間,都是從網易上面轉載的新聞,同時,測試表數據字段類型是比較均勻的,為了更好的測試,表結構如下圖:

數據量:129,991 條記錄
語法分析
1. NTILE() 的分頁方法
NTILE() 方法可以用來分頁,但是應用場景十分的狹窄,并且性能差勁,和 Row_Number() 與 offset fetch 分頁比起來沒有任何優(yōu)勢,也只有在只讀表上面分頁的話,還是比較合適的;雖然不好用,但是還能來分頁的,所以只簡單的介紹下。
語法:
NTILE (integer_expression) OVER ( [ <partition_by_clause> ] < order_by_clause > )
將有序分區(qū)中的行分發(fā)到指定數目的組中。 各個組有編號,編號從一開始。 對于每一個行,NTILE 將返回此行所屬的組的編號。
測試中用到的 Sql 語句 :
set statistics time on
set statistics io on
set statistics profile on;
with #pager as
(
select ID,Title,NTILE(8666) OVER(Order By ID) as pageid from Article_Detail
)
select ID,Title from #pager where pageid=50
set statistics profile on;
其中上述數字中的 8666 是根據 RowCount / Pagesize 計算出來的,不過多介紹,可以自行參考 MSDN的
2. ROW_NUMBER() 的分頁方法
在 Sql Server 2000 之后的版本中,ROW_NUMBER() 這種分頁方式一直都是很不錯的,比起之前的游標分頁,性能好了很多,因為 ROW_NUMBER() 并不會引起全表掃表,但是,語法比較復雜,并且,隨著頁碼的增加,性能也越來越差。
語法 :
ROW_NUMBER ( ) OVER ( [ PARTITION BY value_expression , ... [ n ] ] order_by_clause )
測試中用到的 Sql 語句:
dbcc freeproccache
dbcc dropcleanbuffers
set statistics time on
set statistics io on
set statistics profile on;
with #pager as
(
select ID,Title,ROW_NUMBER() OVER(Order By ID) as rowid from Article_Detail
)
select ID,Title from #pager where rowid between (15 * (50-1)+1) and 15 * 50
set statistics profile off;
3. Offset and Fetch 的分頁方法
語法:
OFFSET { integer_constant | offset_row_count_expression } { ROW | ROWS }
FETCH { FIRST | NEXT } { integer_constant | fetch_row_count_expression } { ROW | ROWS } ONLY
從語法可以看出來 兩個方法 后面不但能接 intege 類型的參數,還能接 表達式的,比如 1*2 +3 之類的,同時, Row 或者 Rows 是不區(qū)分大小寫和單復數的哦
在看測試用的 Sql 語句,真的是簡潔的不能再簡潔了,看兩遍都能記住的語法,分頁可以如此的簡潔:
dbcc freeproccache
dbcc dropcleanbuffers
set statistics time on
set statistics io on
set statistics profile on;
select ID,Title from Article_Detail order by id OFFSET (15 * (50-1)) ROW FETCH NEXT 15 rows only
set statistics profile off;
一句就搞定!
性能比較
1. NTILE() 的執(zhí)行計劃
2. ROW_NUMBER() 的執(zhí)行計劃
3. Offset and Fetch 的 執(zhí)行計劃
綜合比較:
在 Sql Server 2012 里面,分頁方法中,Offset and Fetch 同 ROW_NUMBER() 比較起來,無論是性能還是語法,都是有優(yōu)勢的。
但是性能方面,優(yōu)勢并不是太大,兩者 的 IO 消耗完全相同,只是 在 CPU 方面,Offset and Fetch 方面要好一些,但是不明顯。如果對于一個 每秒都要處理成千上萬條的分頁Sql語句的DB 來說,Offset and Fetch 在CPU 方面的優(yōu)勢會比較明顯的,否則,性能的提升并不明顯。
語法方面 Offset and Fetch 則是十分的簡潔,一句搞定,比起 Row_Number() 好了太多 ~
同是 Offset and Fetch 并不僅僅可以用來分頁哦,具體其他使用,大家可以自行參考 MSDN
準備工作,建立測試表:Article_Detail,主要是用來存放一些文章信息,測試的時間,都是從網易上面轉載的新聞,同時,測試表數據字段類型是比較均勻的,為了更好的測試,表結構如下圖:

內容:

數據量:129,991 條記錄
語法分析
1. NTILE() 的分頁方法
NTILE() 方法可以用來分頁,但是應用場景十分的狹窄,并且性能差勁,和 Row_Number() 與 offset fetch 分頁比起來沒有任何優(yōu)勢,也只有在只讀表上面分頁的話,還是比較合適的;雖然不好用,但是還能來分頁的,所以只簡單的介紹下。
語法:
NTILE (integer_expression) OVER ( [ <partition_by_clause> ] < order_by_clause > )
將有序分區(qū)中的行分發(fā)到指定數目的組中。 各個組有編號,編號從一開始。 對于每一個行,NTILE 將返回此行所屬的組的編號。
測試中用到的 Sql 語句 :
復制代碼 代碼如下:
set statistics time on
set statistics io on
set statistics profile on;
with #pager as
(
select ID,Title,NTILE(8666) OVER(Order By ID) as pageid from Article_Detail
)
select ID,Title from #pager where pageid=50
set statistics profile on;
其中上述數字中的 8666 是根據 RowCount / Pagesize 計算出來的,不過多介紹,可以自行參考 MSDN的
2. ROW_NUMBER() 的分頁方法
在 Sql Server 2000 之后的版本中,ROW_NUMBER() 這種分頁方式一直都是很不錯的,比起之前的游標分頁,性能好了很多,因為 ROW_NUMBER() 并不會引起全表掃表,但是,語法比較復雜,并且,隨著頁碼的增加,性能也越來越差。
語法 :
ROW_NUMBER ( ) OVER ( [ PARTITION BY value_expression , ... [ n ] ] order_by_clause )
測試中用到的 Sql 語句:
復制代碼 代碼如下:
dbcc freeproccache
dbcc dropcleanbuffers
set statistics time on
set statistics io on
set statistics profile on;
with #pager as
(
select ID,Title,ROW_NUMBER() OVER(Order By ID) as rowid from Article_Detail
)
select ID,Title from #pager where rowid between (15 * (50-1)+1) and 15 * 50
set statistics profile off;
3. Offset and Fetch 的分頁方法
語法:
OFFSET { integer_constant | offset_row_count_expression } { ROW | ROWS }
FETCH { FIRST | NEXT } { integer_constant | fetch_row_count_expression } { ROW | ROWS } ONLY
從語法可以看出來 兩個方法 后面不但能接 intege 類型的參數,還能接 表達式的,比如 1*2 +3 之類的,同時, Row 或者 Rows 是不區(qū)分大小寫和單復數的哦
在看測試用的 Sql 語句,真的是簡潔的不能再簡潔了,看兩遍都能記住的語法,分頁可以如此的簡潔:
復制代碼 代碼如下:
dbcc freeproccache
dbcc dropcleanbuffers
set statistics time on
set statistics io on
set statistics profile on;
select ID,Title from Article_Detail order by id OFFSET (15 * (50-1)) ROW FETCH NEXT 15 rows only
set statistics profile off;
一句就搞定!
性能比較
1. NTILE() 的執(zhí)行計劃
2. ROW_NUMBER() 的執(zhí)行計劃
3. Offset and Fetch 的 執(zhí)行計劃
綜合比較:
在 Sql Server 2012 里面,分頁方法中,Offset and Fetch 同 ROW_NUMBER() 比較起來,無論是性能還是語法,都是有優(yōu)勢的。
但是性能方面,優(yōu)勢并不是太大,兩者 的 IO 消耗完全相同,只是 在 CPU 方面,Offset and Fetch 方面要好一些,但是不明顯。如果對于一個 每秒都要處理成千上萬條的分頁Sql語句的DB 來說,Offset and Fetch 在CPU 方面的優(yōu)勢會比較明顯的,否則,性能的提升并不明顯。
語法方面 Offset and Fetch 則是十分的簡潔,一句搞定,比起 Row_Number() 好了太多 ~
同是 Offset and Fetch 并不僅僅可以用來分頁哦,具體其他使用,大家可以自行參考 MSDN
您可能感興趣的文章:
- 高效的SQLSERVER分頁查詢(推薦)
- sqlserver2005使用row_number() over分頁的實現方法
- SQL SERVER 2008 中三種分頁方法與比較
- oracle,mysql,SqlServer三種數據庫的分頁查詢的實例
- 真正高效的SQLSERVER分頁查詢(多種方案)
- SQL Server 分頁查詢存儲過程代碼
- 五種SQL Server分頁存儲過程的方法及性能比較
- sqlserver分頁的兩種寫法分別介紹
- sqlserver 通用分頁存儲過程
- sqlserver 存儲過程分頁(按多條件排序)
- sql server中千萬數量級分頁存儲過程代碼
- sqlserver 高性能分頁實現分析
- SQL Server 分頁查詢通用存儲過程(只做分頁查詢用)
- sql server實現分頁的方法實例分析
相關文章
Sql存儲過程游標循環(huán)的用法及sql如何使用cursor寫一個簡單的循環(huán)
這篇文章主要介紹了循環(huán)和游標在Sql存儲過程中使用及sql如何使用cursor寫一個簡單的循環(huán)的相關資料,需要的朋友可以參考下2015-11-11存儲過程配合UpdateDaset方法批量插入Dataset數據實現代碼
用存儲過程配合UpdateDaset方法批量插入Dataset數據,感興趣的你可不要錯過了哈,希望本文提供知識點可以幫助到你2013-02-02SQL Server實現split函數分割字符串功能及用法示例
這篇文章主要介紹了SQL Server實現split函數分割字符串功能及用法,結合實例形式分析了SQL Server實現split分割字符串的相關技巧與使用方法,需要的朋友可以參考下2016-08-08SQL?Server新特性SequenceNumber用法介紹
這篇文章介紹了SQL?Server新特性SequenceNumber的用法,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2022-02-02sql server把退款總金額拆分到盡量少的多個訂單中詳解
這篇文章主要給大家介紹了關于sql server把退款總金額拆分到盡量少的多個訂單中的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-12-12SQL 判斷給定日期值(或時間段)所在星期的星期一和星期天的日期
最近報表要用到一項功能,需要把數據源根據記錄發(fā)生日期所在的星期序列進行分組。因此就寫了兩個相關SQL Function進行調用。2011-10-10學習SQL語句(強大的group by與select from模式)
本文介紹的是強大的group by使用與利用select from (select from)的模式生成SQL語句的代碼。2011-10-10實例講解sql server排名函數DENSE_RANK的用法
這篇文章主要以一個實例講解了sql server排名函數DENSE_RANK的用法,感興趣的小伙伴們可以參考一下2016-03-03