`

Oracle的rownum原理和使用(分页查询) 

 
阅读更多

要显示12行则可以通过

      select * from dangan where rownum between 1 and 2

Oracle中,要按特定条件查询前N条记录,用个rownum就搞定了。
      select * from emp where rownum <= 5
而且书上也告诫,不能对rownum">",这也就意味着,如果你想用
      select * from emp where rownum > 5
则是失败的。要知道为什么会失败,则需要了解rownum背后的机制:
     1 Oracle executes your query.

     2 Oracle fetches the first row and calls it row number 1.

     3 Have we gotten past row number meets the criteria? If no, then Oracle discards the row, If yes,  then Oracle return the row.

     4 Oracle fetches the next row and advances the row number (to 2, and then to 3, and then to 4, and so forth).

     5 Go to step 3.

了解了原理,就知道rownum>不会成功,因为在第三步的时候查询出的行已经被丢弃,第四步查出来的rownum仍然是1,这样永远也不会成功。

同样道理,rownum如果单独用=,也只有在rownum=1时才有用。

 

对于rownum来说它是oracle系统顺序分配为从查询返回的行的编号,返回的第一行分配的是1,第二行是2,依此类推,这个伪字段可以用于限制查询返回的总行数,而且rownum不能以任何表的名称作为前缀。
 
举例说明:
例如表:student(学生)表,表结构为:
ID
       char(6)      --学号
name
    VARCHAR2(10)   --姓名
create table student (ID char(6), name VARCHAR2(100));
insert into sale values('200001',‘
张一’);
insert into sale values('200002',‘
王二’);
insert into sale values('200003',‘
李三’);
insert into sale values('200004',‘
赵四’);
commit;
(1) rownum
对于等于某值的查询条件
如果希望找到学生表中第一条学生的信息,可以使用rownum=1作为条件。但是想找到学生表中第二条学生的信息,使用rownum=2结果查不到数据。因为rownum都是从1开始,但是1以上的自然数在rownum做等于判断时认为都是false条件,所以无法查到rownum = nn>1的自然数)。
SQL> select rownum,id,name from student where rownum=1;
(可以用在限制返回记录条数的地方,保证不出错,如:隐式游标)
SQL> select rownum,id,name from student where rownum=1;
    ROWNUM ID     NAME
---------- ------ ---------------------------------------------------
         1 200001
张一
SQL> select rownum,id,name from student where rownum =2;
    ROWNUM ID     NAME
---------- ------ ---------------------------------------------------
2rownum对于大于某值的查询条件
  
如果想找到从第二行记录以后的记录,当使用rownum>2是查不出记录的,原因是由于rownum是一个总是从1开始的伪列,Oracle 认为rownum> n(n>1的自然数)这种条件依旧不成立,所以查不到记录
SQL> select rownum,id,name from student where rownum >2;
ROWNUM ID     NAME
---------- ------ ---------------------------------------------------
那如何才能找到第二行以后的记录呀。可以使用以下的子查询方法来解决。注意子查询中的rownum必须要有别名,否则还是不会查出记录来,这是因为rownum不是某个表的列,如果不起别名的话,无法知道rownum是子查询的列还是主查询的列。
SQL>select * from(select rownum no ,id,name from student) where no>2;
        NO ID     NAME
---------- ------ ---------------------------------------------------
         3 200003
李三
         4 200004
赵四
SQL> select * from(select rownum,id,name from student)where rownum>2;
    ROWNUM ID     NAME
---------- ------ ---------------------------------------------------
3rownum对于小于某值的查询条件
如果想找到第三条记录以前的记录,当使用rownum<3是能得到两条记录的。显然rownum对于rownum<n(n>1的自然数)的条件认为是成立的,所以可以找到记录。
SQL> select rownum,id,name from student where rownum <3;
    ROWNUM ID     NAME
---------- ------ ---------------------------------------------------
1 200001
张一
        2 200002
王二
综上几种情况,可能有时候需要查询rownum在某区间的数据,那怎么办呀从上可以看出rownum对小于某值的查询条件是认为true的,rownum对于大于某值的查询条件直接认为是false的,但是可以间接的让它转为认为是true的。那就必须使用子查询。例如要查询rownum在第二行到第三行之间的数据,包括第二行和第三行数据,那么我们只能写以下语句,先让它返回小于等于三的记录行,然后在主查询中判断新的rownum的别名列大于等于二的记录行。但是这样的操作会在大数据集中影响速度。
SQL> select * from (select rownum no,id,name from student where rownum<=3 ) where no >=2;
        NO ID     NAME
---------- ------ ---------------------------------------------------
         2 200002
王二
         3 200003
李三
4rownum和排序
Oracle
中的rownum的是在取数据的时候产生的序号,所以想对指定排序的数据去指定的rowmun行数据就必须注意了。
SQL> select rownum ,id,name from student order by name;
    ROWNUM ID     NAME
---------- ------ ---------------------------------------------------
         3 200003
李三
         2 200002
王二
         1 200001
张一
         4 200004
赵四
可以看出,rownum并不是按照name列来生成的序号。系统是按照记录插入时的顺序给记录排的号,rowid也是顺序分配的。为了解决这个问题,必须使用子查询
SQL> select rownum ,id,name from (select * from student order by name);
    ROWNUM ID     NAME
---------- ------ ---------------------------------------------------
         1 200003
李三
         2 200002
王二
         3 200001
张一
         4 200004
赵四
这样就成了按name排序,并且用rownum标出正确序号(有小到大)

*****************************************

ROWNUM使用的几个注意点

对于 Oracle rownum 问题,很多资料都说不支持>,>=,=,between...and,只能用以上符号(<<=!=),并非说用>, >=,=,between..and 时会提示SQL语法错误,而是经常是查不出一条记录来,还会出现似乎是莫名其妙的结果来,其实您只要理解好了这个 rownum 伪列的意义就不应该感到惊奇,同样是伪列,rownum rowid 可有些不一样,下面以例子说明

假设某个表 t1(c1) 20 条记录

如果用 select rownum,c1 from t1 where rownum < 10, 只要是用小于号,查出来的结果很容易地与一般理解在概念上能达成一致,应该不会有任何疑问的。

可如果用 select rownum,c1 from t1 where rownum > 10 (如果写下这样的查询语句,这时候在您的头脑中应该是想得到表中后面10条记录),你就会发现,显示出来的结果要让您失望了,也许您还会怀疑是不谁删了一 些记录,然后查看记录数,仍然是 20 条啊?那问题是出在哪呢?

先好好理解 rownum 的意义吧。因为ROWNUM是对结果集加的一个伪列,即先查到结果集之后再加上去的一个列 (强调:先要有结果集)。简单的说 rownum 是对符合条件结果的序列号。它总是从1开始排起的。所以你选出的结果不可能没有1,而有其他大于1的值。所以您没办法期望得到下面的结果集:

11 aaaaaaaa
12 bbbbbbb
13 ccccccc
.................

rownum >10 没有记录,因为第一条不满足去掉的话,第二条的ROWNUM又成了1,所以永远没有满足条件的记录。或者可以这样理解:

ROWNUM是一个序列,是oracle数据库从数据文件或缓冲区中读取数据的顺序。它取得第一条记录则rownum值为1,第二条为2,依次类推。如果 你用>,>=,=,between...and这些条件,因为从缓冲区或数据文件中得到的第一条记录的rownum1,则被删除,接着取下 条,可是它的rownum还是1,又被删除,依次类推,便没有了数据。

有了以上从不同方面建立起来的对 rownum 的概念,那我们可以来认识使用 rownum 的几种现像

1. select rownum,c1 from t1 where rownum != 10 为何是返回前9条数据呢?它与 select rownum,c1 from tablename where rownum < 10 返回的结果集是一样的呢?
因为是在查询到结果集后,显示完第 9 条记录后,之后的记录也都是 != 10,或者 >=10,所以只显示前面9条记录。也可以这样理解,rownum 9后的记录的 rownum10,因条件为 !=10,所以去掉,其后记录补上,rownum又是10,也去掉,如果下去也就只会显示前面9条记录了

2. 为什么 rownum >1 时查不到一条记录,而 rownum >0 rownum >=1 却总显示所以的记录
因为 rownum 是在查询到的结果集后加上去的,它总是从1开始

3. 为什么 between 1 and 10 或者 between 0 and 10 能查到结果,而用 between 2 and 10 却得不到结果
原因同上一样,因为 rownum 总是从 1 开始

从上可以看出,任何时候想把 rownum = 1 这条记录抛弃是不对的,它在结果集中是不可或缺的,少了rownum=1 就像空中楼阁一般不能存在,所以你的 rownum 条件要包含到 1

但如果就是想要用 rownum > 10 这种条件的话话就要用嵌套语句, rownum 先生成,然后对他进行查询。
select *
from (selet rownum as rn
t1.* from a where ...)
where rn >10

一般代码中对结果集进行分页就是这么干的。

另外:rowid rownum 虽都被称为伪列,但它们的存在方式是不一样的,rowid 可以说是物理存在的,表示记录在表空间中的唯一位置ID,在DB中唯一。只要记录没被搬动过,rowid是不变的。rowid 相对于表来说又像表中的一般列,所以以 rowid 为条件就不会有 rownum那些情况发生。
另外还要注意:rownum不能以任何基表的名称作为前缀。

 

网上的另一种解释:

Oraclerownum使用
   
今天去一家公司笔试时,又遇到了一道以前曾遇到过的SQL题,是这样的:

    如果想从一个表中以某一字段排序后,取得除了2030行以外的记录,写一条SQL来实现。

 

    想到了用rownumoracle专用的一个伪列。当时写的有点问题,回来后试了一下,不行!

 

    于是重新在网上查了一下,下面是基本的实现:

    假设表结构如下:

 

    Student(sid, sname, age)

 

    SQL:sid升序排序

select * from
(select a.*, rownum rn from

    (select * from strdent order by sid ) a

)
where
rn < 20 or rn > 30;

    如果分页时,想取得第2030条记录可以用以下的方法:

select * from
(select a.*, rownum rn from

    (select * from strdent order by sid ) a where rownum <= 30

)
where rn >= 20;

    这里有一点需要说明,在最内层的查询中,因为是全部的记录,所以在这次查询排序后,每条记录的rownum已经确定,并在第二层的查询中将它作为一个字段记录下来,在最外层的条件中使用。就是说在查询结果中,必须以第一次排序查询时的rownum为基准,统一使用这时候的rownum值,结果才是正确的。如果在第二次加上条件后的查询结果,它还会有一个rownum,但是它的值已经变了,如果用这个值,那么结果也就不是我们想要的了。

    Oraclerownum字段是个比较奇怪的字段。拿一张有26条记录的Test表来举例。
     select * from Test where rownum >=1;
     select * from Test where rownum >=2;
     select * from Test where rownum <= 10;
   
第一条sql查出了26条记录,第二条sql一条记录也没查出。第三条sql查出10条记录。

    导致这个的原因是因为rownum是个虚拟的字段,它是在记录输出的时候逐步产生的。

    对第一条sql,第一条记录的rownum1,满足条件被输出,因此第二条纪录的rownum就变成2,满足条件被输出,依此类推,就把所有纪录都查出来了。
   
对于第二条sql,第一条记录的rownum1,不满足条件没被输出,因此第二条记录的rownum还是1,没满足条件没被输出,依此类推,所有纪录都没能被查出来。

 

 

分享到:
评论

相关推荐

    ROWNUM的使用技巧

    oracle 分页查询 使用ROWNUM技巧及陷阱

    Oracle Rownum的使用与JSP分页显示的实现.pdf

    Oracle Rownum的使用与JSP分页显示的实现.pdf

    oracle rownum 学习

    内涵oracle rownum的详细讲解。

    oracle rownum 使用技术.pdf

    oracle rownum 使用小技巧 里面包含分页

    Oracle中使用Rownum分页详细例子

    主要介绍了Oracle中使用Rownum分页详细例子,本文将分别展示使用rownum伪列和row_number()分析函数来完成Oracle数据分页操作的具体使用方法,需要的朋友可以参考下

    oracle 使用rownum的三种分页方式

    rownum是Oracle数据库中的一个特有关键字,返回的是一个数字代表记录的行号。这篇文章主要介绍了oracle 使用rownum的三种分页方式,需要的朋友可以参考下

    对于 Oracle 的 rownum 问题

    对于 Oracle 的 rownum 问题,很多资料都说不支持&gt;,&gt;=,=,between...and,只能用以上符号(&lt;、、!=),并非说用&gt;,&gt;=,=,between..and 时会提示SQL语法错误,而是经常是查不出一条记录来,还会出现似乎是莫名其妙的结果来...

    oracle的分页查询

    使用数据库时有时我们会需要分页查询,但是在oracle中使用查询条件时又不可以使用大于号&gt;。

    asp.net使用oracle分页存储过程查询数据

    写好oracle的连接字符串和查询语句,调用程序中的方法可以很方便的实现分页功能。该方法中,将参数连接字符串,查询的sql语句,指定每页显示多少行,调用成功后,会返回页数,行数,还有查询的结果数据集。 使用...

    Oracle中实现分页查询的SQL命令

    Oracle中实现分页查询的SQL命令 //curPage是当前页面,pageCount是每页显示行数 //rownum是伪列,相当于表中每一列的标识列(可以理解为行号),需要显式的提取出来并取一个别名

    Oracle中使用伪列rownum实现分页查询

    rownum是查询返回的结果集中行的序号,可以使用它来限制查询返回的行数 从emp表中查询薪水排在前三位的记录 select * from( select * from emp order by sal desc ) where rownum&lt;=3 结果如下: 原创文章 4获赞...

    oracle一条sql语句分页

    oracle 中没有像其他数据库那样用top来实现分页,但oracle提供了rownum 通过它也可以实现分页方法。

    oracle的rownum深入解析

    本人最近在使用oracle的rownum实现分页显示的时候,对rownum做了进一步的分析和研究。现归纳如下,希望能给大家带来收获。 对于rownum来说它是oracle系统顺序分配为从查询返回的行的编号,返回的第一行分配的是1,第...

    Oracle分页查询性能优化代码详解

    对于数据库中表的数据的 Web 显示,如果没有展示顺序的需要,而且因为满足条件的记录如此之多,就不得不对数据进行分页处理。常常用户并不是对所有数据都感兴趣的,或者大部分情况下,他们只看前几页。 通常有以下...

    oracle分页操作

    而另一种方式是去掉查询第二层的WHERE ROWNUM 语句,在查询的最外层控制分页的最小值和最大值。这是,查询语句如下: SELECT * FROM ( SELECT A.*, ROWNUM RN FROM (SELECT * FROM TABLE_NAME) A ) WHERE RN ...

    Oracle使用MyBatis中RowBounds实现分页查询功能

    Oracle中分页查询因为存在伪列rownum,sql语句写起来较为复杂,现在介绍一种通过使用MyBatis中的RowBounds进行分页查询,非常方便。 使用MyBatis中的RowBounds进行分页查询时,不需要在 sql 语句中写 offset,limit...

    简单实例解释Oracle分页查询

    我们可以为了以下目的使用分页查询: 为了精确定位结果集的内容 为了节约内存 为了在有限的页面空间显示适度的数据。 如何实现分页查询 使用Oracle的EMP表 select * from ( select rownum rn, e.* from ( ...

    Oracle分页查询的实例详解

    Oracle分页查询的实例详解 1.Oracle分页查询: SELECT * FROM ( SELECT A.*, ROWNUM RN FROM (SELECT * FROM tab) A WHERE ROWNUM &lt;= 40 ) WHERE RN &gt;= 21; 这个分页比下面的执行时间少,效率高。  2. ...

Global site tag (gtag.js) - Google Analytics