一、基本概念
一个表空间由若干个数据文件组成,表空间的属性可以从dba_tablespaces中获取:
SQL> desc dba_tablespaces;
TABLESPACE_NAME 名称
BLOCK_SIZE 一个blocik的大小,
INITIAL_EXTENT 分配给一个segment的第一个extent的大小,以字节为单位 NEXT_EXTENT,第二个extent的大小等于next的值,以后的next=前一next*(1+pctincrease/100)
MIN_EXTENTS segment 第一次创建时分配的extent数量
MAX_EXTENTS 一个segment最多的extent数量
PCT_INCREASE 见next_extent
MIN_EXTLEN
STATUS 状态 online|offline
CONTENTS permanent|temp|undo
LOGGING 是否记录redo
FORCE_LOGGING yes|no对表空间里的segment记录redo,即使该segment的logging属性logging
EXTENT_MANAGEMENT local | dictionary
ALLOCATION_TYPE system|uniform extent的分配类型:系统分配,或者是统一
PLUGGED_IN
SEGMENT_SPACE_MANAGEMENT 段管理 manual|auto 手动后自动
DEF_TAB_COMPRESSION
RETENTION
BIGFILE 是否大文件(10g)
二、数据文件的组织和段的空间分配
用得最多的是extent本地管理,segment自动管理的表空间。这种情况下,每个段使用位图来记录空间的使用情况,位图分为三个级别,L0,L1,L2 .其中第一个L0块就是段头,L0-L1-L2是一种树形结构。
BMB : bitmap block
以下是一个数据文件可能的例子:
|
Lmt使用(数据文件的前8个 block) |
|||||||
Segment 1 extent 0 |
First level bmb (L1) |
second lev el bmb (L2) |
Segment header(L0) |
Data block |
Data block |
Data block |
Data block |
Data block |
Segment 1 extent 1 |
Data block |
Data block |
Data block |
Data block |
Data block |
Data block |
Data block |
Data block |
Segment 1 extent 2 |
L2 |
Data block |
Data block |
Data block |
Data block |
Data block |
Data block |
Data block |
Segment 2 extent 0 |
L1 |
L2 |
Segment header(L0) |
Data block |
Data block |
Data block |
Data block |
Data block |
Segment 2 extent 1 |
Data block |
Data block |
Data block |
Data block |
Data block |
Data block |
Data block |
Data block |
…… |
…… |
…… |
…… |
…… |
…… |
…… |
…… |
…… |
每格为1个block,block size为8k,每个extent有8个block
每个数据文件最前面的8个块,是LMT用于管理extent分配的,每一个segmen的前三个block分别是bmb L1,bmb L2,和段头信息(L0),L1还会出现在其它extent中,但不是每个extent都会有L1。
三、通过直接路径插入和间接路径插入的对比来了解段的空间分配(在ITPUB晶晶小妹文章中的例子基础上)
直接路径插入与间接路径插入
(一)准备
1.创建两个表。
SQL> create table aa(id number(4),name varchar2(5));
SQL> create table bb(id number(4),name varchar2(5));
2.aa表中插入数据
SQL> insert into aa values(1,'aa');
SQL> insert into aa values(2,'bb');
SQL> insert into aa values(3,'cc');
SQL> insert into aa values(4,'dd');
SQL> commit;
3.查看这些数据所在的数据块,可以看到这四条记录都位于文件4的463块上
SQL> select dbms_rowid.rowid_relative-fno(rowid),dbms_rowid.rowid_block_number(rowid) from aa;
DBMS_ROWID.ROWID_RELATIVE_FNO(ROWID) BMS_ROWID.ROWID_BLOCK_NUMBER(ROWID)
4 463
4 463
4 463
4 463
4.清空缓冲区,通过重新启动数据库来实现。使用alter system flush buffer cache不能完整清空。
SQL> shutdown immediate;
SQL> startup
5.查看缓冲区内aa、bb的数据块,可以看到这时缓冲区没有这两个表的数据块
SQL> select file#,block# from v$bh where objd=(select object_id from user_object
s where object_name='AA');
未选定行
SQL> select file#,block# from v$bh where objd=(select object_id from user_object
s where object_name='BB');
未选定行
5.再次查看aa记录所在的数据块
SQL> select dbms_rowid.rowid_relative_fno(rowid),dbms_rowid.rowid_block_number(owid) from aa;
DBMS_ROWID.ROWID_RELATIVE_FNO(ROWID) BMS_ROWID.ROWID_BLOCK_NUMBER(ROWID)
4 463
4 463
4 463
4 463
6.由于上一个查询操作,现在可以看到aa的数据块463和其它信息块已经读入缓存,bb也可类似查询。
SQL> select file#,block# from v$bh where objd=(select object_id from user_object
s where object_name='AA');
FILE# BLOCK#
4 461
4 464
4 459
4 459
4 462
4 460
4 463
SQL> select file#,block# from v$bh where objd=(select object_id from user_object
s where object_name='BB');
FILE# BLOCK#
4 467
4 467
7.通过alter system dump datafile 4 block 467,可以看到这是bb的块头,且高水位为468,使用了3个块。
...
frmt: 0x02 chkval: 0x6632 type: 0x23=PAGETABLE SEGMENT HEADER
Hex dump of block: st=0, typ_found=1
Extent Header:: spare1: 0 spare2: 0 #extents: 1 #blocks: 8
last map 0x00000000 #maps: 0 offset: 2716
Highwater:: 0x010001d4 ext#: 0 blk#: 3 ext size: 8
(二)直接路径插入
1、直接插入
SQL> insert into BB select * from aa;
SQL> commit;
2、查询,可以看到,bb的数据块468并没有在缓存中:
SQL> select file#,block# from v$bh where objd=(select object_id from user_object
s where object_name='BB');
FILE# BLOCK#
4 467
4 467
4 465
SQL> select dbms_rowid.rowid_relative_fno(rowid),dbms_rowid.rowid_block_number(r
owid) from BB;
DBMS_ROWID.ROWID_RELATIVE_FNO(ROWID) BMS_ROWID.ROWID_BLOCK_NUMBER(ROWID)
4 468
3、465块通过dump出来看,是‘FIRST LEVEL BITMAP BLOCK’,也就是段自动管理使用的位图块L1,这是因为插入数据后要更新位图块信息。而467块显示该segment已用块数变为4;
465 dump:
...
frmt: 0x02 chkval: 0x0000 type: 0x20=FIRST LEVEL BITMAP BLOCK
Hex dump of block: st=0, typ_found=1
467 dump:
...
frmt: 0x02 chkval: 0x0000 type: 0x23=PAGETABLE SEGMENT HEADER
Hex dump of block: st=0, typ_found=1
Extent Header:: spare1: 0 spare2: 0 #extents: 1 #blocks: 8
last map 0x00000000 #maps: 0 offset: 2716
Highwater:: 0x010001d5 ext#: 0 blk#: 4 ext size: 8
(二)间接路径插入
1、间接插入
SQL> insert into BB select * from aa;
SQL> commit;
2、查看
SQL> select file#,block# from v$bh where objd=(select object_id from user_object
s where object_name='BB');
FILE# BLOCK#
4 469
4 472
4 467
4 467
4 470
4 465
4 468
4 471
4 466
SQL> select dbms_rowid.rowid_relative_fno(rowid),dbms_rowid.rowid_block_number(rowid) from BB;
DBMS_ROWID.ROWID_RELATIVE_FNO(ROWID) BMS_ROWID.ROWID_BLOCK_NUMBER(ROWID)
4 468
4 468
4 468
4 468
4 472
4 472
4 472
4 472
可以看到,bb的数据块468,472都已经在缓存中
3、 dump 467
...
Highwater:: 0x010001d9 ext#: 0 blk#: 8 ext size: 8
...
#blocks in seg. hdr's freelists: 0
#blocks below: 8
mapblk 0x00000000 offset: 0
Level 1 BMB for High HWM block: 0x010001d1
Level 1 BMB for Low HWM block: 0x010001d1
Segment Type: 1 nl2: 1 blksz: 8192 fbsz: 0
L2 Array start offset: 0x00001434
First Level 3 BMB: 0x00000000
L2 Hint for inserts: 0x010001d2
Last Level 1 BMB: 0x010001d1
Last Level II BMB: 0x010001d2
Last Level III BMB: 0x00000000
Map Header:: next 0x00000000 #extents: 1 obj#: 52657 flag: 0x10000000
Inc # 0
Extent Map
0x010001d1 length: 8
Auxillary Map
Extent 0 : L1 dba: 0x010001d1 Data dba: 0x010001d4
Second Level Bitmap block DBAs
高水位变为473,已用块变为8,bmb L1 地址 0x010001d1(file 4,block 465), bmb L2 地址 0x010001d2(file 4,block 466),数据块开始位置0x010001d4(file 4,block 468)