ORACLE中用rownum分頁並排序的SQL語句
以前分頁習慣用這樣的SQL語句:
1
2
3
|
select
* from ( select
t.*,rownum row_num from
mytable t order
by t.id) b where
b.row_num between
1 and 10 |
結果發現由於該語句會先生成rownum 後執行order by 子句,因而排序結果根本不對,後來在GOOGLE上搜到一篇文章,原來多套一層select 就能很好的解決該問題,特此記錄,語句如下:
1
2
3
4
5
|
select
* from ( select
a.*,rownum row_num from ( select
* from
mytable t order
by t.id desc ) a ) b
where
b.row_num between
1 and 10 |
==== 參考 ====
http://yangtingkun.itpub.net/post/468/100278
Oracle的分頁查詢語句基本上可以按照本文給出的格式來進行套用。分頁查詢格式:
1
2
3
4
5
6
7
|
SELECT
* FROM ( SELECT
A.*, ROWNUM RN FROM
( SELECT
* FROM TABLE_NAME) A WHERE
ROWNUM <= 40 ) WHERE
RN >= 21 |
其中最內層的查詢SELECT * FROM TABLE_NAME表示不進行翻頁的原始查詢語句。ROWNUM <= 40和RN >= 21控制分頁查詢的每頁的範圍。
上面給出的這個分頁查詢語句,在大多數情況擁有較高的效率。分頁的目的就是控制輸出結果集大小,將結果儘快的返回。在上面的分頁查詢語句中,這種考慮主要體現在WHERE ROWNUM <= 40這句上。
選擇第21 到40條記錄存在兩種方法,一種是上面例子中展示的在查詢的第二層通過ROWNUM <= 40來控制最大值,在查詢的最外層控制最小值。而另一種方式是去掉查詢第二層的WHERE ROWNUM <= 40語句,在查詢的最外層控制分頁的最小值和最大值。這是,查詢語句如下:
1
2
3
4
5
6
|
SELECT
* FROM ( SELECT
A.*, ROWNUM RN FROM
( SELECT
* FROM TABLE_NAME) A ) WHERE
RN BETWEEN 21
AND 40 |
對比這兩種寫法,絕大多數的情況下,第一個查詢的效率比第二個高得多。
http://www.javaworld.com.tw/jute/post/view?bid=21&id=52022&sty=1&tpg=1&age=-1
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
SELECT
* FROM ( SELECT
A.*, rownum r FROM -- 這裡的SQL可以改成你真正要執行的SQL ( SELECT
* FROM
Articles ORDER
BY PubTime DESC ) A -- -- 用上面的SQL得回來的集合,使用rownum去比對,這樣rownum就會從這
個集合的第一筆資料開始往下計算,所以這邊是抓取前100筆 WHERE
rownum <= 100 ) B
-- B集合總共有A集合和r(rownum)的資料,這裡是抓取大於第90筆的 WHERE
r > |