分页查询有用的两个东东。

Rowid和Rownum对于数据库开发人员来说基本很少用到,因为在企业数据库开发中大多都是进行数据批处理,但是对于其他数据库人员来说还是会用到的。
rowid和rownum都是虚列,但含义完全不同。rowid是物理地址,用于定位oracle中具体数据的物理存储位置,而rownum则是sql的输出结果排序。通俗的讲:rowid是相对不变的,rownum会变化,尤其是使用order by的时候。

ROWID:
1、为什么使用ROWID
   ORACLE把ROWID作为B-树和其内部算法标示ROW的唯一标示。在 ORACLE8以前的版本中,ROWID标示FILE、BLOCK、ROW NUMBER,只用一个数字代表FILE号。在ORACLE8中,一个DATAFILE有两个数字代表:   一个绝对值,是整个数据库唯一的。可以看DBA_DATA_FILES中的FILE_ID。   一个相对值,在TABLESPACE中是唯一的,可以看DBA_DATA_FILES中的RELATIVE_FNO。

  新的ROWID使用相对值,所以必须存放SEGMENT的标示,否则就会混淆。所以ORACLE8在ROWID中加入对象的SEGMENT号,用来标示TABLE或者PARTITION。

2、ROWID的结构
   使用base-64代码,包括a-z,A-Z,0-9,+,-。一共18位。   1-6位:代表OBJECT   7-9位:文件相对值   10-15:文件中的BLOCK   16-18:BLOCK中的SLOT值

3、TABLESPACE-Relative寻址方式
   使用的是TABLESPACE-Relative寻址方式,多个文件可以有相同 的相对值,因为它   们属于不同的TABLESPACE,所以不能从新的ROWID得到绝对地址,但是这没有问题   ,因为当要处理某个OBJECT时,已经能确定它属于哪个TABLESAPCE了。在TABLESPACE中,文件相对值是唯一的,所以ROWID还 是可以唯一标示一个OBJECT。TABLESPACE-Relative寻址方式是ORACLE8中支持超大数据库的关键技术。

4、DATA OBJECT NUMBER
   DATA OBJECT NUMBER用于指示SEGMENT(部分/片段),所有SEGMENT都有DATA OBJECT NUMBER,存放在每个DATA BLOCK(块/区)中,而且不重复。

  最开始的时候,DBA_OBJECTS.OBJECT_ID=DBA_OBJECTS.DATA-OBJECT_ID,但是在上述情况下 DATA-OBJECT_ID会在如下情况下增加   TRUNCATE TABLE   MOVE PARTITION   ORACLE会检查ROWID中的DATA OBJECT NUMBER和BLOCK中的DATA OBJECT NUMBER,保证他们之间的版本是一致的。   ORACLE也使用DATA OBJECT NUMBER以确保ROLLBACK的纪录和最新的SEGMENT纪录一致。   要注意的是DATA OBJECT NUMBER不是OBJECT 的标志

5、RESTRICTED ROWID
   ORACLE7的ROWID格式是   1-8位:BLOCK NUMBER   9-12位:ROW NUMBER   13-16位:FILE NUMBER   ORACLE8支持短的、旧格式的ROWID,作用是   对NOPARTITION TABLE的INDEX ENTRY   对PARTITION TABLE的LOCAL INDEX ENTRY   ROW Piece CHain pointer   受限ROWID的内部存放是6BYTE,   4BYTE=DATA BLOCK NUMBER   2BYTE=ROW NUMBER   这就是说,INDEX ENTRY使用6BYTE存放该ROWID,这对大多数INDEX足够了。但是这种短ROWID不能使用在PATITION TABLE的GLOBAL INDEX上,因为PARTITION可能跨TABLESPACE。显示这种ROWID依然是18位的

6、扩展的ROWID
 ORACLE在内部存放时候是10 BYTE,包括(DATA OBJECT NUMBER,DATA BLOCK NU   MBER,ROW NUMBER)   ORACLE8使用扩展的ROWID:   PARTITION TABLE 的GLOBAL INDEX   SERVER 算法   扩展的ROWID在SELECT时,依然是18位的显示,存放在ROWID字段中。

7、在ORACLE8中使用ORACLE7的ROWID
 从ORACLE8的DB中查询ORACLE7的ROWID时候,ROWID返回的 是ORACLE7的格式,也可以用在WHERE语句中。   从ORACLE7的DB中查询ORACLE8的ROWID时候,ROWID返回的是ORACLE8的格式,也可以用在WHERE语句中,但是不能存放 在ROWID字段中。但是你要用DBMS_ROWID 包来解释之。   如果包含扩展的ORACLE8 ROWID,这不能把ORACLE8的数据IMPORT到ORACLE7中。从ORACLE7中可以IMPORT到ORACLE8中。

8、APPLICATION的移植问题
 一般程序的移植应该没有问题。只有在下面情况下才考虑移植问题:   application使用了rowid   table包括ROWID类型的字段   如果程序有如下情况,必须使用DBMS_ROWID包:   自己组合ROWID   自己分解ROWID   如果仅仅是传递ROWID到变量、或者仅仅做为一个整体使用,则可以不受影响。

9、数据的移植问题
 无论使用EXPORT/IMPORT还使用移植工具,ORACLE7中的ROWID字段到了ORACLE8中就自动扩展。如果在某个字段内容中包含ROWID,则必须手工用DBMS_ROWID包来转换。

10、DBMS_ROWID包
   由$ORACLE_HOME/rdbms/admin/dbmsutil.sql创建,其实在 catproc.sql中包含着。提供处理ROWID的一些函数。   ROWID_CREATE   ROWID_INFO   ROWID_TYPE   ROWID_OBJECT   ROWID_RELATIVE_FNO   ROWID_BLOCK_NUMBER   ROWID_TO_ABSOLUTE_FNO   ROWID_TO_EXTENDED   ROWID_TO_RESTRICTED   ROWID_VERIFY

    DBMS_ROWID.ROWID_TO_EXTENDED   (old_rowid in ROWID,   schema_name in varchar2,   object_name in varchar2,   conversion_type in number   )   RETURN ROWID;   转换受限rowid到扩展rowid,用于转换旧的ROWID到ORACLE8的格式。

    DBMS_ROWID.ROWID_TO_RESTRICTED 转换扩展的ROWID到受限的ROWID。

  DBMS_ROWID.ROWID_VERIFY   判断一个受限的ROWID是否可以转换到扩展的格式

    DBMS_ROWID.ROW_INFO   用于解释ROWID,可以得到DATA OBJECT NUMBER,RELATIVE FILE NUMBER,BLOCK NUMBER和ROW NUMBER。

  DBMS_ROWID.CREATE   生成ROWID。

ROWNUM:

在Oracle中,要按特定条件查询前N条记录,用个rownum就搞定了。 select * from emp whererownum<= 5 而且书上也告诫,不能对rownum用">",这也就意味着,如果你想用 select * from emp whererownum> 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= n(n>1的自然数)。 SQL> selectrownum,id,name from student whererownum=1;(可以用在限制返回记录条数的地方,保证不出错,如:隐式游标) SQL> selectrownum,id,name from student whererownum=1;    ROWNUMID     NAME ---------- ------ ---------------------------------------------------          1 200001 张一 SQL> selectrownum,id,name from student whererownum=2;    ROWNUMID     NAME ---------- ------ --------------------------------------------------- (2)rownum对于大于某值的查询条件    如果想找到从第二行记录以后的记录,当使用rownum>2是查不出记录的,原因是由于rownum是一个总是从1开始的伪列,Oracle 认为rownum> n(n>1的自然数)这种条件依旧不成立,所以查不到记录 SQL> selectrownum,id,name from student whererownum>2;ROWNUMID     NAME ---------- ------ --------------------------------------------------- 那如何才能找到第二行以后的记录呀。可以使用以下的子查询方法来解决。注意子查询中的rownum必须要有别名,否则还是不会查出记录来,这是因为 rownum不是某个表的列,如果不起别名的话,无法知道rownum是子查询的列还是主查询的列。 SQL>select * from(selectrownumno ,id,name from student) where no>2;         NO ID     NAME ---------- ------ ---------------------------------------------------          3 200003 李三          4 200004 赵四 SQL> select * from(selectrownum,id,name from student)whererownum>2;    ROWNUMID     NAME ---------- ------ --------------------------------------------------- (3)rownum对于小于某值的查询条件如果想找到第三条记录以前的记录,当使用rownum<3是能得到两条记录的。显然rownum对于 rownum<n((n>1的自然数)的条件认为是成立的,所以可以找到记录。 SQL> selectrownum,id,name from student whererownum<3;    ROWNUMID     NAME ---------- ------ --------------------------------------------------- 1 200001 张一         2 200002 王二综上几种情况,可能有时候需要查询rownum在某区间的数据,那怎么办呀从上可以看出rownum对小于某值的查询条件是人为true 的,rownum对于大于某值的查询条件直接认为是false的,但是可以间接的让它转为认为是true的。那就必须使用子查询。例如要查询rownum 在第二行到第三行之间的数据,包括第二行和第三行数据,那么我们只能写以下语句,先让它返回小于等于三的记录行,然后在主查询中判断新的rownum的别 名列大于等于二的记录行。但是这样的操作会在大数据集中影响速度。 SQL> select * from (selectrownumno,id,name from student whererownum<=3 ) where no >=2;         NO ID     NAME ---------- ------ ---------------------------------------------------          2 200002 王二          3 200003 李三(4)rownum和排序 Oracle中的rownum的是在取数据的时候产生的序号,所以想对指定排序的数据去指定的rowmun行数据就必须注意了。 SQL> selectrownum,id,name from student order by name;    ROWNUMID     NAME ---------- ------ ---------------------------------------------------          3 200003 李三          2 200002 王二          1 200001 张一          4 200004 赵四可以看出,rownum并不是按照name列来生成的序号。系统是按照记录插入时的顺序给记录排的号,rowid也是顺序分配的。为了解决这个问题, 必须使用子查询 SQL> selectrownum,id,name from (select * from student order by name);    ROWNUMID     NAME ---------- ------ ---------------------------------------------------          1 200003 李三          2 200002 王二          3 200001 张一          4 200004 赵四这样就成了按name排序,并且用rownum标出正确序号(小到大)