Oracle SQL篇(三)Oracle ROWNUM 与TOP N分析
含义解释:
1、rownum是oracle为从查询返回的行的编号,返回的第一行分配的是1,第二行是2,依此类推。这是一个伪列,可以用于限制查询返回的总行数。
2、rownum不能以任何基表的名称作为前缀。
对于ROWNUM来说,通常我们可以使用的比较符是<和<=,不能单独的使用=、>、>=等比较运算符,其实我们可以这样简单的
理解,oracle是找到第一条的记录添加序号1之后,才可以知道谁是第二条记录,然后添加序号2,以此类推。所以对于等于来说,是可以有例外的,就是
rownum=1。
我们来看几个简单的演示:
scott@DB01> create table demo as select
demono,ename,sal,comm,deptno from demo;
Table created.
scott@DB01> select rownum,t.* from demo t;
ROWNUM
demoNO
ENAME
SAL
COMM
DEPTNO
---------- ---------- ---------- ---------- ----------
----------
1
7369
SMITH
800
20
2
7499
ALLEN
1600
300
30
3
7521
WARD
1250
500
30
4
7566
JONES
2975
20
5
7654
MARTIN
1250
1400
30
6
7698
BLAKE
2850
30
7
7782
CLARK
2450
10
8
7788
SCOTT
3000
20
9
7839
KING
5000
10
10
7844
TURNER
1500
0
30
11
7876
ADAMS
1100
20
12
7900
JAMES
950
30
13
7902
FORD
3000
20
14
7934
MILLER
1300
10
14 rows selected.
scott@DB01> select rownum,demono,ename,sal from demo
where rownum<=3;
ROWNUM
demoNO
ENAME
SAL
---------- ---------- ---------- ----------
1
7369
SMITH
800
2
7499
ALLEN
1600
3
7521
WARD
1250
scott@DB01> select rownum,demono,ename,sal from demo
where rownum<3;
ROWNUM
demoNO
ENAME
SAL
---------- ---------- ---------- ----------
1
7369
SMITH
800
2
7499
ALLEN
1600
scott@DB01> select rownum,demono,ename,sal from demo
where rownum=1;
ROWNUM
demoNO
ENAME
SAL
---------- ---------- ---------- ----------
1
7369
SMITH
800
scott@DB01> select rownum,demono,ename,sal from demo
where rownum=3;
no rows selected
scott@DB01> select rownum,demono,ename,sal from demo
where rownum>3;
no rows selected
如果我们想要查询结果集中的某一段范围的记录,比如5-10条的记录,该如何查询呢?很多开发人员把这样的需求称为分页
scott@DB01> select rownum,demono,ename,sal from demo
where rownum between 5 and 10;
no rows selected
上面是一个错误的例子,我们来看正确的写法,这里我们使用到了集合运算符minus(减法运算)
scott@DB01> select rownum,demono,ename,sal from demo
where rownum<=10
2
minus
3
select rownum,demono,ename,sal from demo where
rownum<=4;
ROWNUM
demoNO
ENAME
SAL
---------- ---------- ---------- ----------
5
7654
MARTIN
1250
6
7698
BLAKE
2850
7
7782
CLARK
2450
8
7788
SCOTT
3000
9
7839
KING
5000
10
7844
TURNER
1500
6 rows selected.
如果我们有这样一个需求,找到员工demo表中,薪水最高的前三名,如何来实现呢?在sql server中有标准的top
n分析语句,不过不要放到oracle里来使用,不同的数据库还是有区别的。当然了,也许你会认为这个问题有歧义,是前三个人呢?还是薪水排在最高3位的人?因为薪水有可能是相同的,在这里我们就找前三个人,看下面的语句:
scott@DB01> select rownum,demono,ename,sal
2> from demo
3> where rownum<=3
4> order by sal desc;
ROWNUM
demoNO
ENAME
SAL
---------- ---------- ---------- ----------
2
7499
ALLEN
1600
3
7521
WARD
1250
1
7369
SMITH
800
这个语句从表面上来看好像是正确的,从demo表里查询数据,排序,最后利用rownum返回前三个人,但是我们看语句的执行结果显然是不正确的。对于oracle的语句,我们在执行的时候遵循top-down的顺序,或者我们可以说,语句按照顺序来执行。
当然也有个别例外:
scott@DB01> select deptno,sum(sal) from demo
2
group by deptno
3
having sum(sal)>=10000;
DEPTNO SUM(SAL)
---------- ----------
20
10875
scott@DB01> select deptno,sum(sal) from demo
2
having sum(sal)>=10000
3
group by deptno;
DEPTNO SUM(SAL)
---------- ----------
20
10875
对于前面排名的语句当然是有问题的。当第2行语句执行后,拿到表中所有的数据,第3行语句紧接着执行,就把最前面的三条记录取出来了(rownum是对查询结果添加序号),这个时候,再做排序,当然拿到的就是对前面三条记录排序的结果,如上所示。
正确的思路应该是,先做排序,再做条件筛选,也就是4行子句要在3行之前运行,如何来实现呢?在这里oracle借助了子查询,用oracle的标准表述叫做内联视图(inline
view),当然整个的sql就是oracle的top N分析语句的写法,我们来看例子:
scott@DB01> select rownum
rank,t.*
2
from (select demono,ename,sal from demo order by sal desc) t
3
where rownum<=3;
RANK
demoNO
ENAME
SAL
---------- ---------- ---------- ----------
1
7839
KING
5000
2
7788
SCOTT
3000
3
7902
FORD
3000
在这里,oracle其实对内部子查询做了优化处理,我们通常认为,简单子查询是内层查询先执行,然后传递结果给外层查询,然后外层查询再执行。
但是对于这个例子,如果demo表数据量很大的话,那么内层排序需要花的时间就会非常多。而实际上呢,oracle会知道外层查询需要的记录数,如本例中是3,
oracle在对内层查询排序时,并不是对demo表中的14条记录做完全的排序,根据算法,他只要找到sal最高的3条就可以了,其余的11条记录是没必要排序的,这就大大的节省了语句的执行时间。
如果想要得到排序后的某段数据,我们可以通过嵌套的方法来实现:
scott@DB01> select
t1.*
2
from (select rownum rank,t.*
from (select demono,ename,sal from demo order by sal desc) t)
t1
3
where rank>=3 and rank<=7;
RANK
demoNO
ENAME
SAL
---------- ---------- ---------- ----------
3
7788
SCOTT
3000
4
7566
JONES
2975
5
7698
BLAKE
2850
6
7782
CLARK
2450
7
7499
ALLEN
1600
注:在前面两个例子中,我们使用到了oracle的top N分析,不过都是对整张表,或者整个结果集来说的。其实oracle
对于类似的操作,提供了一套函数,我们称之为分析函数,分析函数对于数据做统计和分析是非常有帮助的,我们在下面只是举一个简单的小例子,如果你感兴趣可以看看
scott@DB01> select * from demo;
demoNO
ENAME
SAL
COMM
DEPTNO
---------- ---------- ---------- ---------- ----------
7369
SMITH
800
20
7499
ALLEN
1600
300
30
7521
WARD
1250
500
30
7566
JONES
2975
20
7654
MARTIN
1250
1400
30
7698
BLAKE
2850
30
7782
CLARK
2450
10
7788
SCOTT
3000
20
7839
KING
5000
10
7844
TURNER
1500
0
30
7876
ADAMS
1100
20
7900
JAMES
950
30
7902
FORD
3000
20
7934
MILLER
1300
10
14 rows selected.
scott@DB01> break on deptno skip 1
scott@DB01> select *
2 from (select deptno,
3
ename,
4
sal,
5
dense_rank() over(partition by deptno order by sal desc) dr
6
from demo
7
)
8 where
dr<=3
9 order by deptno;
DEPTNO
ENAME
SAL
DR
---------- ---------- ---------- ----------
10
KING
5000
1
CLARK
2450
2
MILLER
1300
3
20
SCOTT
3000
1
FORD
3000
1
JONES
2975
2
ADAMS
1100
3
30
BLAKE
2850
1
ALLEN
1600
2
TURNER
1500
3
10 rows selected.
scott@DB01> select * from (
2
select deptno,
3
ename,
4
sal,
5
row_number() over(partition by deptno order by sal desc) dr
6
from demo
7
)
8 where
dr<=3
9 order by deptno;
DEPTNO
ENAME
SAL
DR
---------- ---------- ---------- ----------
10
KING
5000
1
CLARK
2450
2
MILLER
1300
3
20
SCOTT
3000
1
FORD
3000
2
JONES
2975
3
30
BLAKE
2850
1
ALLEN
1600
2
TURNER
1500
3
Oracle SQL篇(三)Oracle ROWNUM 与TOP N分析的更多相关文章
- Oracle SQL篇(一)null值之初体验
从我第一次正式的写sql语句到现在,已经超过10年的时间了.我写报表,做统计分析和财务对账,我一点点的接触oracle数据库,并尝试深入了解.这条路,一走就是10年,从充满热情,到开始厌 ...
- 安装并使用Oracle SQL Developer访问Oracle
---问题 如何安装并使用Oracle SQL Developer访问Oracle. ---步骤 Oracle SQL Developer是Oracle官方出品的免费图形化开发工具,相对SQL*Plu ...
- Oracle SQL Developer,Oracle 开发工具之toad、SQL Developer、PL/SQL Developer等比较
参考: oracle 的几个开发工具比较 因Oracle几乎是中大型商业企业数据的首选,所以比较一下常用与Oracle的工具. Oracle SQL Developer 免费,一般开发使用足矣,常用. ...
- oracle学习篇三:SQL查询
select * from emp; --1.找出部门30的员工select * from emp where deptno = 30; --2.列出所有办事员(CLERK)的姓名,变化和部门编号se ...
- Oracle SQL篇(二)oracle自连接操作
oracle 的自连接(self join)操作 对于oracle数据库来说,目前支持两套语法,一套是oracle自己的sql语法,一套是通行标准的SQL99语法,那么对于oracle的连接操 ...
- Oracle SQL篇(四)group by 分组与分组的加强 rollup
分组操作group by 和分组的强化(rollup) 分组操作和分组函数的使用,对于编写SQL语句的人来说,是最基本的概念. 我们来看下面的例子: 在这里我们使用员工表EMP scott@D ...
- Oracle Sql Developer 连接oracle
PL/Sql 初次使用需要配置文件内容,对于我这种Oracle新手来说各种配置有点凌乱,所以果断选择Sql Developer. 选择它是因为初次使用的时候它不用想PL/Sql那样配置文件,而只需要添 ...
- (转) Oracle SQL优化必要的全表扫描思路分析
大多数情况下,我们需要避免SQL在查询时进行全表扫描(FTS),但是对于必须需要进行全表扫描的情况,也可以进行一些优化处理. 即使全表扫描是检索所需数据的唯一可行方法,仍然有多种方法来提升查询性能.优 ...
- Oracle SQL tuning 步骤
Oracle SQL tuning 步骤 SQL是的全称是Structured Query Language(结构化查询语言).SQL是一个在80年代中期被使用的工业标准数据库查询语言.不要把SQL语 ...
随机推荐
- C语言的复合文字
假设需要向一个带有int型参量的函数传递一个值,这时可以传递一个int型常量,也可以传递一个int型的变量.在C99标准之前,数组参数情况于现在不一样,没有所谓的数组常量可供传递,而在C99中增加了复 ...
- c#操作word文档之简历导出
前言 1.写这个功能之前,我得说说微软的这个类库,用着真苦逼!是他让我有程序猿,攻城尸的感觉了.首先这个类库,从没接触过,方法与属性都不懂,还没有提示.神啊,我做这功能真是一步一卡,很潇洒啊. 2.这 ...
- 创建Java项目报错处理
好久没用Eclipse编写Java程序了,今天创建一个Java项目的时候,老报错,错误信息如下: Implicit super constructor Object() is undefined fo ...
- Deep Learning(深度学习)相关网站
Deep Learning(深度学习) ufldl的2个教程(这个没得说,入门绝对的好教程,Ng的,逻辑清晰有练习):一 ufldl的2个教程(这个没得说,入门绝对的好教程,Ng的,逻辑清晰有练习): ...
- Deep Learning(深度学习)学习笔记整理系列之(八)
Deep Learning(深度学习)学习笔记整理系列 zouxy09@qq.com http://blog.csdn.net/zouxy09 作者:Zouxy version 1.0 2013-04 ...
- CXF 调用C#.net的WebService
原文链接:http://hi.baidu.com/pengfeiiw/blog/item/3203e29065aa3a8aa977a4d0.html 1.编写C#.net的WebService Ser ...
- mysql 的not null 与 null的区别(转,恍然大悟)
相信很多用了mysql很久的人,对这两个字段属性的概念还不是很清楚,一般会有以下疑问: 1.我字段类型是not null,为什么我可以插入空值 2.为毛not null的效率比null高 3.判断字段 ...
- 常用笔记: 与VBS当中的Mid()类似的substr()小记
VBS当中有Mid函数,一般形式为:Mid(str,start,len) 对应于JS就类似于:str.substr(start,len) 不过区别的是:VBS中start从1开始,而JS从0开始. ...
- Java Web 部署到Tomcat
1.在conf目录中,新建Catalina(注意大小写)\localhost目录,在该目录中新建一个xml文件,名字可以随意取,只要和当前文件中的文件名不重复就行了,该xml文件的内容为: <C ...
- codeforces 401D. Roman and Numbers 数位dp
题目链接 给出一个<1e18的数, 求将他的各个位的数字交换后, 能整除m的数的个数. 用状态压缩记录哪个位置的数字已经被使用了, 具体看代码. #include<bits/stdc++. ...