📄 oracle开发技巧-4.htm
字号:
<body bgcolor="#000000">
<p><font color="#009900"><strong>oracle8的ROWID结构</strong></font></p>
<p><font color="#FFFFFF">1、为什么使用ROWID <br>
ORACLE把ROWID作为B-树和其内部算法标示ROW的唯一标示。在ORACLE8以前的版本中,ROWID标示FILE、BLOCK、ROW NUMBER,只用一个数字代表FILE号。在ORACLE8中,一个DATAFILE有两个数字代表:
<br>
一个绝对值,是整个数据库唯一的。可以看DBA_DATA_FILES中的FILE_ID。 <br>
一个相对值,在TABLESPACE中是唯一的,可以看DBA_DATA_FILES中的RELATIVE_FNO。 </font></p>
<font color="#FFFFFF">
<p>新的ROWID使用相对值,所以必须存放SEGMENT的标示,否则就会混淆。所以ORACLE8在ROWID中加入对象的SEGMENT号,用来标示TABLE或者PARTITION。
</p>
<p>2、ROWID的结构 <br>
使用base-64代码,包括a-z,A-Z,0-9,+,-。一共18位。 <br>
1-6位:代表OBJECT <br>
7-9位:文件相对值 <br>
10-15:文件中的BLOCK <br>
16-18:BLOCK中的SLOT值 </p>
<p>3、TABLESPACE-Relative寻址方式 <br>
使用的是TABLESPACE-Relative寻址方式,多个文件可以有相同的相对值,因为它 <br>
们属于不同的TABLESPACE,所以不能从新的ROWID得到绝对地址,但是这没有问题 <br>
,因为当要处理某个OBJECT时,已经能确定它属于哪个TABLESAPCE了。在TABLES <br>
PACE中,文件相对值是唯一的,所以ROWID还是可以唯一标示一个OBJECT。TABLE <br>
SPACE-Relative寻址方式是ORACLE8中支持超大数据库的关键技术。 </p>
<p>4、DATA OBJECT NUMBER <br>
DATA OBJECT NUMBER用于指示SEGMENT,所有SEGMENT都有DATA OBJECT NUMBER,存放在每个DATA BLOCK中,而且不重复。
</p>
<p>最开始的时候,DBA_OBJECTS.OBJECT_ID=DBA_OBJECTS.DATA-OBJECT_ID,但是在上述情况下DATA-OBJECT_ID会在如下情况下增加
<br>
TRUNCATE TABLE <br>
MOVE PARTITION <br>
ORACLE会检查ROWID中的DATA OBJECT NUMBER和BLOCK中的DATA OBJECT NUMBER,保证他们之间的版本是一致的。
<br>
ORACLE也使用DATA OBJECT NUMBER以确保ROLLBACK的纪录和最新的SEGMENT纪录一致。 <br>
要注意的是DATA OBJECT NUMBER不是OBJECT 的标志 </p>
<p><br>
5、RESTRICTED ROWID <br>
ORACLE7的ROWID格式是 <br>
1-8位:BLOCK NUMBER <br>
9-12位:ROW NUMBER <br>
13-16位:FILE NUMBER <br>
ORACLE8支持短的、旧格式的ROWID,作用是 <br>
对NOPARTITION TABLE的INDEX ENTRY <br>
对PARTITION TABLE的LOCAL INDEX ENTRY <br>
ROW Piece CHain pointer <br>
受限ROWID的内部存放是6BYTE, <br>
4BYTE=DATA BLOCK NUMBER <br>
2BYTE=ROW NUMBER <br>
这就是说,INDEX ENTRY使用6BYTE存放该ROWID,这对大多数INDEX足够了。但是这种短ROWID不能使用在PATITION TABLE的GLOBAL
INDEX上,因为PARTITION可能跨TABLESPACE。显示这种ROWID依然是18位的 </p>
<p>6、扩展的ROWID <br>
ORACLE在内部存放时候是10 BYTE,包括(DATA OBJECT NUMBER,DATA BLOCK NU <br>
MBER,ROW NUMBER) <br>
ORACLE8使用扩展的ROWID: <br>
PARTITION TABLE 的GLOBAL INDEX <br>
SERVER 算法 <br>
扩展的ROWID在SELECT时,依然是18位的显示,存放在ROWID字段中。 </p>
<p>7、在ORACLE8中使用ORACLE7的ROWID <br>
从ORACLE8的DB中查询ORACLE7的ROWID时候,ROWID返回的是ORACLE7的格式,也可以用在WHERE语句中。 <br>
从ORACLE7的DB中查询ORACLE8的ROWID时候,ROWID返回的是ORACLE8的格式,也可以用在WHERE语句中,但是不能存放在ROWID字段中。但是你要用DBMS_ROWID
包来解释之。 <br>
如果包含扩展的ORACLE8 ROWID,这不能把ORACLE8的数据IMPORT到ORACLE7中。从ORACLE7中可以IMPORT到ORACLE8中。
</p>
<p>8、APPLICATION的移植问题 <br>
一般程序的移植应该没有问题。只有在下面情况下才考虑移植问题: <br>
application使用了rowid <br>
table包括ROWID类型的字段 <br>
如果程序有如下情况,必须使用DBMS_ROWID包: <br>
自己组合ROWID <br>
自己分解ROWID <br>
如果仅仅是传递ROWID到变量、或者仅仅做为一个整体使用,则可以不受影响。 </p>
<p><br>
9、数据的移植问题 <br>
无论使用EXPORT/IMPORT还使用移植工具,ORACLE7中的ROWID字段到了ORACLE8中就自动扩展。如果在某个字段内容中包含ROWID,则必须手工用DBMS_ROWID包来转换。
</p>
<p>10、DBMS_ROWID包 <br>
由$ORACLE_HOME/rdbms/admin/dbmsutil.sql创建,其实在catproc.sql中包含着。提供处理ROWID的一些函数。
<br>
ROWID_CREATE <br>
ROWID_INFO <br>
ROWID_TYPE <br>
ROWID_OBJECT <br>
ROWID_RELATIVE_FNO <br>
ROWID_BLOCK_NUMBER <br>
ROWID_TO_ABSOLUTE_FNO <br>
ROWID_TO_EXTENDED <br>
ROWID_TO_RESTRICTED <br>
ROWID_VERIFY </p>
</font>
<p></p>
<p><font color="#FFFFFF">DBMS_ROWID.ROWID_TO_EXTENDED <br>
(old_rowid in ROWID, <br>
schema_name in varchar2, <br>
object_name in varchar2, <br>
conversion_type in number <br>
) <br>
RETURN ROWID; <br>
转换受限rowid到扩展rowid,用于转换旧的ROWID到ORACLE8的格式。 </font></p>
<p><font color="#FFFFFF"><br>
DBMS_ROWID.ROWID_TO_RESTRICTED 转换扩展的ROWID到受限的ROWID。 </font></p>
<p><font color="#FFFFFF">DBMS_ROWID.ROWID_VERIFY <br>
判断一个受限的ROWID是否可以转换到扩展的格式 </font></p>
<p><font color="#FFFFFF"><br>
DBMS_ROWID.ROW_INFO <br>
用于解释ROWID,可以得到DATA OBJECT NUMBER,RELATIVE FILE NUMBER,BLOCK NUMBER和ROW NUMBER。
</font></p>
<p><font color="#FFFFFF">DBMS_ROWID.CREATE <br>
生成ROWID。 <br>
</font></p>
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -