一、ROWID的格式:
第二部分3位表示:该行数据所在的相对数据文件的id;
第三部分6位表示:该数据行所在的数据块的编号;
第四部分3位表示:该行数据的行的编号;
索引就是保存了rowid后三个部分的信息。索引是物理存在的,而rowid是伪列。所以索引可以用来快速地定位到数据行。
data_object_id
下面以SAKILA数据库的ACTOR表为例 这里我们要注意将 data_object_id 与 object_id 区分开来,前者是oracle为它的每一个对象唯一分配的id,而后者与表ACTOR对应的“段”有关,是存放表tt的段的id,也就是与存放表tt中数据的物理位置有关:select owner,object_id,data_object_id,status from dba_objects where object_name='ACTOR';
alter table ACTOR move tablespace users;
select owner,object_id,data_object_id,status from dba_objects where object_name='ACTOR';
我们看到当表ACTOR move到了users表空间时,段发生了改变,物理位置发生了变化,从而 DATA_OBJECT_ID 也发生了变化。我们知道表是存放在“表段”中的,索引是存放在“索引段”中的。DATA_OBJECT_ID就是表示存放数据的“数据段对象的id”
相对文件编码
关于相对文件编码和绝对文件编号:相对文件id是指相对于表空间,在表空间唯一,绝对文件是指相当于全局数据库而言的,全局唯一;select file_name,file_id,relative_fno from dba_data_files;
rowid采用64进制来编码
编码方法是:A~Z表示0到25;a~z表示26到51;0~9表示52到61;+表示62;/表示63;刚好64个字符。Base64编码表
码值 | 字符 | 码值 | 字符 | 码值 | 字符 | 码值 | 字符 | |||
---|---|---|---|---|---|---|---|---|---|---|
0 | A | 16 | Q | 32 | g | 48 | w | |||
1 | B | 17 | R | 33 | h | 49 | x | |||
2 | C | 18 | S | 34 | i | 50 | y | |||
3 | D | 19 | T | 35 | j | 51 | z | |||
4 | E | 20 | U | 36 | k | 52 | 0 | |||
5 | F | 21 | V | 37 | l | 53 | 1 | |||
6 | G | 22 | W | 38 | m | 54 | 2 | |||
7 | H | 23 | X | 39 | n | 55 | 3 | |||
8 | I | 24 | Y | 40 | o | 56 | 4 | |||
9 | J | 25 | Z | 41 | p | 57 | 5 | |||
10 | K | 26 | a | 42 | q | 58 | 6 | |||
11 | L | 27 | b | 43 | r | 59 | 7 | |||
12 | M | 28 | c | 44 | s | 60 | 8 | |||
13 | N | 29 | d | 45 | t | 61 | 9 | |||
14 | O | 30 | e | 46 | u | 62 | + | |||
15 | P | 31 | f | 47 | v | 63 | / |
二、使用rowid访问数据的执行计划
SELECT t.*, ''||t.ROWID FROM "SAKILA"."ACTOR" t;
EXPLAIN PLAN FOR
select * FROM ACTOR where rowid='AAAYEVAAJAAAACrAAA';
select * from table(DBMS_XPLAN.DISPLAY)
三、例子:如何从rowid计算得到obj#,rfile#,block#,row#
我们演示一下具体的计算方法:SELECT t.*, ''||t.ROWID FROM "SAKILA"."ACTOR" t;
码值 | 字符 |
0 | A |
24 | Y |
4 | E |
21 | V |
select owner,object_id,data_object_id,status from dba_objects where object_name='ACTOR';
select file_name,tablespace_name from dba_data_files;
表ACTOR中的第一行数据存放的行的编号为AAAYEVAAJAAAACrAAA 中的 AAA,显然值为0,即第一行。 我们也可以通过Oracle提供的存储过程来计算出上面的值:
SELECT
dbms_rowid.rowid_object (ROWID) data_object_id,
dbms_rowid.rowid_relative_fno (ROWID) relative_fno,
dbms_rowid.rowid_block_number (ROWID) block_no,
dbms_rowid.rowid_row_number (ROWID) row_no
FROM
ACTOR;