ROWNUM偽列
ROWNUM
是一個(gè)偽列,它是根據(jù)每次查詢的結(jié)果動(dòng)態(tài)生成的一列遞增編號(hào),表示 Oracle 從表中選擇該行的順序,選擇的第一行ROWNUM
為1,第二行ROWNUM
為2,以此類推。
注意1:
ROWNUM
偽列是在WHERE
子句之前生成的,就是說(shuō)它并不是在執(zhí)行了WHERE
子句過(guò)濾之后再對(duì)數(shù)據(jù)編號(hào)
比如在執(zhí)行WHERE
子句,結(jié)果數(shù)據(jù)是這樣的:
id | name | age | ROWNUM |
---|---|---|---|
0001 | mary | 18 | 1 |
0002 | mike | 20 | 2 |
0003 | john | 19 | 3 |
0004 | echoo | 16 | 4 |
0005 | susy | 18 | 5 |
0006 | kitty | 21 | 6 |
這時(shí)候的ROWNUM
是一列遞增排列的、完整的編號(hào)
然后如果執(zhí)行一個(gè)WHERE
子句:WHERE age >18
那數(shù)據(jù)就變成了:
id | name | age | ROWNUM |
---|---|---|---|
0002 | mike | 20 | 2 |
0003 | john | 19 | 3 |
0006 | kitty | 21 | 6 |
ROWNUM
出現(xiàn)了斷層,不連續(xù)了
如果后面再執(zhí)行一個(gè)ORDER BY age DESC
就變成這樣了:
id | name | age | ROWNUM |
---|---|---|---|
0006 | kitty | 21 | 6 |
0002 | mike | 20 | 2 |
0003 | john | 19 | 3 |
ROWNUM
不按順序排了
所以在利用ROWNUM
偽列來(lái)對(duì)結(jié)果集做限制、過(guò)濾、排序、分頁(yè)等操作的時(shí)候一定要注意這個(gè)點(diǎn),不然很容易錯(cuò)亂;
注意2:
ROWNUM
是一行一行賦值的,只有上一行數(shù)據(jù)被選擇成功,下一行才會(huì)遞增!而且 select 語(yǔ)句也是一行一行選擇的,每 select 一行數(shù)據(jù)就要進(jìn)行 where 條件判斷。
比如有這樣一個(gè)employees表:
id | name | age |
---|---|---|
0001 | mary | 18 |
0002 | mike | 20 |
0003 | john | 19 |
0004 | kitty | 16 |
0005 | susy | 18 |
0006 | echoo | 21 |
對(duì)這個(gè)表執(zhí)行這樣一個(gè)SQL:
SELECT * FROM employees WHERE ROWNUM > 1;
這句SQL的預(yù)期為取出除第一條數(shù)據(jù)外的所有數(shù)據(jù),但是執(zhí)行的結(jié)果是一條都選不出來(lái),來(lái)看執(zhí)行過(guò)程:
① select 出的第一條數(shù)據(jù)為
id | name | age |
---|---|---|
0001 | mary | 18 |
② ROWNUM
給這條數(shù)據(jù)賦值,因?yàn)槭堑谝粭l數(shù)據(jù),所以從 1 開始,賦值完是這樣的:
id | name | age | ROWNUM |
---|---|---|---|
0001 | mary | 18 | 1 |
③ 進(jìn)行 WHERE ROWNUM > 1
條件判斷,1>1不滿足條件,所以第一條數(shù)據(jù)被過(guò)濾掉
④ select 第二條數(shù)據(jù)
id | name | age |
---|---|---|
0002 | mike | 20 |
⑤ ROWNUM
給這條數(shù)據(jù)賦值,因?yàn)樯弦粭l數(shù)據(jù)被過(guò)濾掉了,所以還是從 1 開始,賦值完是這樣的:
id | name | age | ROWNUM |
---|---|---|---|
0002 | mike | 20 | 1 |
⑥ 進(jìn)行 WHERE ROWNUM > 1
條件判斷,1>1不滿足條件,所以這數(shù)據(jù)也被過(guò)濾掉
⑦ ·············
一直如此循環(huán)直到結(jié)束,都沒(méi)有符合條件的數(shù)據(jù),所以一條數(shù)據(jù)都選不出來(lái)!
例1:取前10條數(shù)據(jù)
用ROWNUM
來(lái)限制查詢返回的行數(shù),如下例所示:
SELECT * FROM employees WHERE ROWNUM < 11;
WHERE ROWNUM < 11
表示返回查詢數(shù)據(jù)的前10條;
例2:取排序后的前10條數(shù)據(jù)
不能夠像下面這樣直接在WHERE
子句后簡(jiǎn)單的加上ORDER BY
子句了
SELECT * FROM employees WHERE ROWNUM < 11 ORDER BY age;
因?yàn)檫@里的意思是先執(zhí)行WHERE
子句選出ROWNUM
1~10的數(shù)據(jù),然后再進(jìn)行排列,和我們預(yù)想不符。我們要的是按年齡排序后的前10條數(shù)據(jù)。
所以應(yīng)該這樣寫:
SELECT *
FROM (SELECT * FROM employees ORDER BY employee_id)
WHERE ROWNUM < 11;
這里的意思就是先執(zhí)行排序,然后對(duì)排完序的結(jié)果集用ROWNUM
偽列按順序編號(hào),然后取其中ROWNUM
為1~10的那10條數(shù)據(jù);文章來(lái)源:http://www.zghlxwxcb.cn/news/detail-653850.html
例3:分頁(yè)
SELECT *
FROM( SELECT temp_table.*,ROWNUM AS rn
FROM (SELECT * FROM employees ORDER BY employee_id) temp_table
) result_table
WHERE result_table.rn BETWEEN 起始行數(shù) AND 結(jié)尾行數(shù)
通過(guò)嵌套查詢的方式,把動(dòng)態(tài)的ROWNUM
偽列變成固定的列rn
,然后再用rn
列進(jìn)行分頁(yè);文章來(lái)源地址http://www.zghlxwxcb.cn/news/detail-653850.html
到了這里,關(guān)于Oracle/PL/SQL奇技淫巧之ROWNUM偽列的文章就介紹完了。如果您還想了解更多內(nèi)容,請(qǐng)?jiān)谟疑辖撬阉鱐OY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持TOY模板網(wǎng)!