1.使用T_BASE_PROVINCE表作為測(cè)試演示使用
專(zhuān)注于為中小企業(yè)提供網(wǎng)站設(shè)計(jì)制作、成都網(wǎng)站設(shè)計(jì)服務(wù),電腦端+手機(jī)端+微信端的三站合一,更高效的管理,為中小企業(yè)金山免費(fèi)做網(wǎng)站提供優(yōu)質(zhì)的服務(wù)。我們立足成都,凝聚了一批互聯(lián)網(wǎng)行業(yè)人才,有力地推動(dòng)了近1000家企業(yè)的穩(wěn)健成長(zhǎng),幫助中小企業(yè)通過(guò)網(wǎng)站建設(shè)實(shí)現(xiàn)規(guī)模擴(kuò)充和轉(zhuǎn)變。
2.查詢下總共數(shù)據(jù)量selectcount(*)fromT_BASE_PROVINCE,在分頁(yè)的時(shí)候,一般會(huì)顯示總頁(yè)數(shù),需要先查詢總數(shù)據(jù)量得到總頁(yè)數(shù),總頁(yè)數(shù)=總量/每頁(yè)顯示記錄數(shù)。
3.前面的測(cè)試數(shù)據(jù)初始化完成之后,查詢前20條大概有什么樣的數(shù)據(jù)。
4.含orderby排序,多一層嵌套,因?yàn)閛rderby在select之后執(zhí)行,不在里面寫(xiě)的話可能會(huì)出現(xiàn)不是預(yù)期的排序結(jié)果。
如以上回答未能解決問(wèn)題請(qǐng)看:
一種是利用相反的。
使用minus,即中文的意思就是減去。
一種是利用Oracle的rownum,這個(gè)是Oracle查詢自動(dòng)返回的序號(hào),一般不顯示,但是可以通過(guò)selectrownumfrom[表名],可以看到,是從1到當(dāng)前的記錄總數(shù)。??
在Oracle中 用SQL來(lái)實(shí)現(xiàn)分頁(yè)有很多種實(shí)現(xiàn)方式 但有些語(yǔ)句可能并不是很通用 只能用在一些特殊場(chǎng)景之中
以下介紹三種比較通用的實(shí)現(xiàn)方案 在以下各種實(shí)現(xiàn)中 ROWNUM是一個(gè)最核心的關(guān)鍵詞 在查詢時(shí)他是一個(gè)虛擬的列 取值為 到記錄總數(shù)的序號(hào)
首先來(lái)介紹我們工作中最常使用的一種實(shí)現(xiàn)方式
SELECT *
FROM (SELECT ROW_ * ROWNUM ROWNUM_
FROM (SELECT *
FROM TABLE
WHERE TABLE _ID = XX
ORDER BY GMT_CREATE DESC) ROW_
WHERE ROWNUM = )
WHERE ROWNUM_ = ;
其中最內(nèi)層的查詢SELECT為不進(jìn)行翻頁(yè)的原始查詢語(yǔ)句 可以用自己的任意Select SQL替換 ROWNUM = 和ROWNUM = 控制分頁(yè)查詢的每頁(yè)的范圍
分頁(yè)的目的就是控制輸出結(jié)果集大小 將結(jié)果盡快的返回 上面的SQL語(yǔ)句在大多數(shù)情況擁有較高的效率 主要體現(xiàn)在WHERE ROWNUM = 這句上 這樣就控制了查詢過(guò)程中的最大記錄數(shù)
上面例子中展示的在查詢的第二層通過(guò)ROWNUM = 來(lái)控制最大值 在查詢的最外層控制最小值 而另一種方式是去掉查詢第二層的WHERE ROWNUM = 語(yǔ)句 在查詢的最外層控制分頁(yè)的最小值和最大值 此時(shí)SQL語(yǔ)句如下 也就是要介紹的第二種實(shí)現(xiàn)方式
SELECT *
FROM (SELECT A * ROWNUM RN
FROM (SELECT *
FROM TABLE
WHERE TABLE _ID = XX
ORDER BY GMT_CREATE DESC) A)
WHERE RN BEEEN AND ;
由于Oracle可以將外層的查詢條件推到內(nèi)層查詢中 以提高內(nèi)層查詢的執(zhí)行效率 但不能跨越多層
對(duì)于第一個(gè)查詢語(yǔ)句 第二層的查詢條件WHERE ROWNUM = 就可以被Oracle推入到內(nèi)層查詢中 這樣Oracle查詢的結(jié)果一旦超過(guò)了ROWNUM限制條件 就終止查詢將結(jié)果返回了
而 第二個(gè)查詢語(yǔ)句 由于查詢條件BEEEN AND 是存在于查詢的第三層 而Oracle無(wú)法將第三層的查詢條件推到最內(nèi)層(即使推到最內(nèi)層也沒(méi)有意義 因?yàn)樽顑?nèi)層查詢不知道RN代表什么) 因此 對(duì)于第二個(gè)查詢語(yǔ)句 Oracle最內(nèi)層返回給中間層的是所有滿足條件的數(shù)據(jù) 而中間層返回給最外層的也是所有數(shù)據(jù) 數(shù)據(jù)的過(guò)濾在最外層完成 顯然這個(gè)效率要比第一個(gè)查詢低得多
以上兩種方案完全是通過(guò)ROWNUM來(lái)完成 下面一種則采用ROWID和ROWNUM相結(jié)合的方式 SQL語(yǔ)句如下
SELECT *
FROM (SELECT RID
FROM (SELECT R RID ROWNUM LINENUM
FROM (SELECT ROWID RID
FROM TABLE
WHERE TABLE _ID = XX
ORDER BY GMT_CREATE DESC) R
WHERE ROWNUM = )
WHERE LINENUM = ) T
TABLE T
WHERE T RID = T ROWID;
從語(yǔ)句上看 共有 層Select嵌套查詢 最內(nèi)層為可替換的不分頁(yè)原始SQL語(yǔ)句 但是他查詢的字段只有ROWID 而沒(méi)有任何待查詢的實(shí)際表字段 具體查詢實(shí)際字段值是在最外層實(shí)現(xiàn)的
這種方式的原理大致為 首先通過(guò)ROWNUM查詢到分頁(yè)之后的 條實(shí)際返回記錄的ROWID 最后通過(guò)ROWID將最終返回字段值查詢出來(lái)并返回
和前面兩種實(shí)現(xiàn)方式相比 該SQL的實(shí)現(xiàn)方式更加繁瑣 通用性也不是非常好 因?yàn)橐獙⒃嫉牟樵冋Z(yǔ)句分成兩部分(查詢字段在最外層 表及其查詢條件在最內(nèi)層)
但這種實(shí)現(xiàn)在特定場(chǎng)景下還是有優(yōu)勢(shì)的 比如我們經(jīng)常要翻頁(yè)到很后面 比如 條記錄中我們經(jīng)常需要查 及其以后的數(shù)據(jù) 此時(shí)該方案效率可能要比前面的高
因?yàn)榍懊娴姆桨钢惺峭ㄟ^(guò)ROWNUM = 來(lái)控制的 這樣就需要查詢出 條數(shù)據(jù) 然后取最后 之間的數(shù)據(jù) 而這個(gè)方案直接通過(guò)ROWID取需要的那 條數(shù)據(jù)
從不斷向后翻頁(yè)這個(gè)角度來(lái)看 第一種實(shí)現(xiàn)方案的成本會(huì)越來(lái)越高 基本上是線性增長(zhǎng) 而第三種方案的成本則不會(huì)像前者那樣快速 他的增長(zhǎng)只體現(xiàn)在通過(guò)查詢條件讀取ROWID的部分
當(dāng)然 除了以上提了這些方案 我們還可以用以下的SQL來(lái)實(shí)現(xiàn)
SELECT *
FROM TABLE
WHERE TABLE _ID NOT IN
(SELECT TABLE _ID FROM TABLE WHERE ROWNUM = )
AND ROWNUM = ;
SELECT *
FROM TABLE
WHERE ROWNUM =
MINUS
SELECT * FROM TABLE WHERE ROWNUM = ;
………………
注意 當(dāng)ROWNUM作為查詢條件時(shí) 他是在order by之前執(zhí)行 所以要特別小心
比如我們想查詢TABLE 中按TABLE _ID倒序排列的前 條記錄不能用如下的SQL來(lái)完成
lishixinzhi/Article/program/Oracle/201311/11198
在Oracle中有一個(gè)方法rownum用來(lái)查詢第一行到第n行的內(nèi)容,但沒(méi)有一個(gè)合適的方法若查詢第x行到第y行的內(nèi)容,而在實(shí)際應(yīng)用中卻經(jīng)常需要查詢第x行到第y行的內(nèi)容,這時(shí)我們就需要使用rownum和子表查詢等內(nèi)容來(lái)進(jìn)行查詢,因?yàn)檫@一塊內(nèi)容屬于Oracle總的常用部分所以專(zhuān)門(mén)在此介紹。
在Oralce中有一個(gè)偽列rownum,其在創(chuàng)建表的時(shí)候就存在了卻不顯示,若要使用這個(gè)列可以直接調(diào)用即可,也可以對(duì)這個(gè)列添加別名來(lái)調(diào)用。
rownum只能用于顯示小于某行的數(shù)據(jù)即第一行開(kāi)始到你要查詢到的那一行為止的數(shù)據(jù)。
在Oracle把查詢第幾行到第幾行的操作稱(chēng)為分頁(yè),其具體操作是通過(guò)子查詢等操作完成。
select 列名 from (select 表名.*,rownum rn from 表名)表名 ?where rn操作;
思考如下:
1.選擇所有內(nèi)容
select * from emp;
2.顯示rownum
select e.*,rownum rn from(select * from emp)e;
這一步可以精簡(jiǎn)為下面形式,但某些情況只能用上面那種
select emp.*,rownum rn from emp;
3.查詢
select * from(select e.*,rownum rn from (select * from emp)e);
4.其他變化
在某些時(shí)候我們需要先對(duì)表的內(nèi)容進(jìn)行排序,隨后查詢第x行到第y行的內(nèi)容,這個(gè)時(shí)候有一個(gè)需要注意的點(diǎn)是rownum是在表產(chǎn)生的時(shí)候產(chǎn)生的偽列,所以使用排序會(huì)連著rownum的值進(jìn)行排序,從而達(dá)不到想要的效果。
為了解決上述這個(gè)問(wèn)題,我們需要使用子表查詢即先排好序,再在新表之中顯示rownum來(lái)規(guī)避這個(gè)問(wèn)題。
考慮到排序的問(wèn)題,所以在上方第二步的時(shí)候使用第一種方法即select e.*,rownum rn from(select * from emp)e;,在內(nèi)表select * from emp中進(jìn)行排序可以完成在亂序中找到第x行到第y行的效果。