Oracle 中关于超过253列内部数据的存储(一)

2014-11-24 18:29:53 · 作者: · 浏览: 0

建表语句
declare
v_sql varchar2(32767) ;
begin
v_sql := 'create table test ( ' ;
for i in 1..500 loop
v_sql := v_sql || 'name'||i||' varchar2(2000) ,' ;
end loop ;
v_sql := substr(v_sql,1,length(v_sql)-1) ;
v_sql := v_sql || ');' ;
dbms_output.put_line(v_sql) ;
end ;
复制粘贴创建表


_dex@DAVID> desc test
Name Null Type
----------------------------------------------------- -------- ------------------------------------
NAME1 VARCHAR2(2000)
NAME2 VARCHAR2(2000)
NAME3 VARCHAR2(2000)
.....
NAME500 VARCHAR2(2000)


_dex@DAVID> insert into test (name500) values (lpad(1,300,'d')) ;


1 row created.



_dex@DAVID> select
2 length(t.name500),
3 dbms_rowid.rowid_relative_fno(t.rowid) as "FNO#",
4 dbms_rowid.rowid_block_number(t.rowid) as "BLK#",
5 dbms_rowid.rowid_row_number(t.rowid) as "ROW#"
6 from dex.test t
7 /


LENGTH(T.NAME500) FNO# BLK# ROW#
----------------- ---------- ---------- ----------
300 4 925 1
1 4 925 3
300 4 925 5



使用bbed查看


首先看row#为1长度为300的数据
BBED> set dba 4,925
DBA 0x0100039d (16778141 4,925)


BBED> p kdbr
sb2 kdbr[0] @142 7504
sb2 kdbr[1] @144 7250
sb2 kdbr[2] @146 6991
sb2 kdbr[3] @148 6737
sb2 kdbr[4] @150 6177
sb2 kdbr[5] @152 5923


BBED> p *kdbr[1]
rowdata[1327]
-------------
ub1 rowdata[1327] @7374 0x28


<16 bytes per line>


这些是row piece header
2801f501 00039d00 00


1个字节的flag=28
16进制28=二进制00101000=--H-F---=head of rowpiece + first data piece
1个字节的lb(itl slot)=01
1个字节的cc column count=f5=245 (因为最后一列name500不为空,所以null都需要使用ff来表示,也就是说需要存储500列的数据)


4个字节的dba
01 00039d=使用dbms_utility.DATA_BLOCK_ADDRESS_FILE以及dbms_utility.DATA_BLOCK_ADDRESS_BLOCK可以以此查询到文件以及block编号
具体:
select dbms_utility.DATA_BLOCK_ADDRESS_FILE(to_number('0100039d','xxxxxxxxxx')) file#,
dbms_utility.DATA_BLOCK_ADDRESS_BLOCK(to_number('0100039d','xxxxxxxxxx')) block#
from dual;


_sys@DAVID> /


FILE# BLOCK#
---------- ----------
4 925
也就是datafile 4 block 925 就在本块内


2个字节的row#
00 00
表示kdbr[0]
剩下的就是表示null的ff了


BBED> p *kdbr[1]
rowdata[1327]
-------------
ub1 rowdata[1327] @7374 0x28


BBED> p *kdbr[0]
rowdata[1581]
-------------
ub1 rowdata[1581] @7628 0x04


7628-7374=254字节(从offset7374开始到7628结束,不包括7628)
其中
9个字节的row header
所以存放row data 的空间为254-9=245字节
因为有245列的null值并且最后一列不为空,所以null都使用ff表示(一个字节),一共消耗了245字节。


我们再来看它的last data piece


BBED> p *kdbr[0]
rowdata[1581]
-------------
ub1 rowdata[1581] @7628 0x04



BBED> dump /v offset 7628 count 1000
File: /u01/apps/oracle/oradata/david/users01.dbf (4)
Block: 925 Offsets: 7628 to 8191 Dba:0x0100039d
-------------------------------------------------------
0401ffff ffffffff ffffffff ffffffff l ................
ffffffff ffffffff ffffffff ffffffff l ................
ffffffff ffffffff ffffffff ffffffff l ................
ffffffff ffffffff ffffffff ffffffff l ................
ffffffff ffffffff ffffffff ffffffff l ................
ffffffff ffffffff ffffffff ffffffff l ................
ffffffff ffffffff ffffffff ffffffff l ................
ffffffff ffffffff ffffffff ffffffff l .....