创建表时考虑列的顺序
时间:2007-12-23 来源:不详 作者:迈克DB
创建一个表时表中列的顺序在某些程度上对性能会有一定的影响.(表中的列有数据)
Oracle对行数据的存储结构ROWHEADER(行头)和COLUMNDATA(列数据).ROWHEADER存储的信息是一个FLAGBYTE,一个LOCKBYTE和COLUMN
COUNT.COLUMNDATA包含COLUMNLENGTH和COLUMNDATA
关于这些我们可以DUMP个表做一下测试
createtestasselect*fromdba_objects;
selectheader_file,header_blockfromdba_segmentswhereowner=’TEST’andsegment_name=’TEST’;
HEADER_FILEHEADER_BLOCK
---------------------------------------------------
131179
altersystemdumpdatafile13block1180
得出来的文件在UDUMP中.我们查看如下信息
block_row_dump:
tab0,row0,@0x1f20
tl:96fb:--H-FL--lb:0x0cc:13--------rowheader信息.
col0:[3]535953-------------------COLUMNDATA
fb:--H-FL--是FLAGBYTE.
fbFlagByte:
K=ClusterKey(FlagsmaychangemeaningifthisissettoshowHASHcluster)
C=Clustertablemember
H=Headpieceofrow
D=Deletedrow
F=Firstdatapiece
L=Lastdatapiece
P=Firstcolumncontinuesfrompreviouspiece
N=Lastcolumncontinuesinnextpiece copyright dedecms
lb:0x0-----------LOCKBYTE,锁信息
cc:13------------COLUMNCOUNT
col0---------第一列
[3]-------------COLUMNLENGTH
535953---------实际数据
这里一些信息可以参考grassbell写的文章<<偷窥Datablock的物理结构>>.对每一个列,在每一个列数据前都含有列长度.在做查询时,查询行中某个
列的值,Oracle首先做的是检查这些相关列的长度位.这个操作比较快而且效率较高.但是假如反复频繁的这样子做还是会带来性能方面的影响.
下面的例子中创建了一个有10列的表并插入数据.
先设置DB_BLOCK_SIZE=2K(用参数设置,在这里设置为这个只是为了测试方便)
SQL>createtablesmall(
2n0number,
3n1number,
4n2number,
5n3number,
6n4number,
7n5number,
8n6number,
9n7number,
10n8number,
11n9number
12)pctfree0;
Tablecreated.
SQL>begin
2foriin1..78loop
3insertintosmallvalues(0,0,0,0,0,0,0,0,0,0);
4endloop;
5end;
6/
PL/SQLproceduresuccessfullycompleted.
SQL>settimingon
SQL>declare
2nnumber;
3begin
4foriin1..1000000loop
5selectsum(n0)intonfromsmall; 内容来自dedecms
6endloop;
7end;
8/
PL/SQLproceduresuccessfullycompleted.
Elapsed:00:07:437.30
SQL>declare
2nnumber;
3begin
4foriin1..1000000loop
5selectsum(n9)intonfromsmall;
6endloop;
7end;
8/
PL/SQLproceduresuccessfullycompleted.
Elapsed:00:08:482.13
从上面的例子很明显可以看到在一个表中做查询时,查询的数据和列的属性都是一样的,但是所查的列位于第一列时查询速度比在第10快了差不多
10%.所以在建表的时候规则就是根据应用将表中经常访问的列放面前面.建表时一般都有一个PRIMARYKEY的列,像这种属性的列一般我们直接访
问的并不多.所以我们一般不放在第一列.关于这个其实假如我们有注重到的话,Oracle本身字典内表也是这样子的.
descdba_objects看看.或者其他的表可以试试.
还有另外一个要考虑的列的位置的就是列中含有较多的NULL值时所要放的位置.
Oracle存储NULL值时,一行中某个列存在NULL值,而这一列的后面的列中存在有数据(非NULL),则Oracle会分配1byte来存放NULL.假如这一列的后面
没有列或者都是NULL值时.这一列和后面的NULL值Oracle都不做存储.列信息也不存储.这一点可以看以下例子.
上一篇:PB动态报表格式自由定义的实现 下一篇:DB2数据库的备份和恢复
文章评论
共有位Admini5网友发表了评论 查看完整内容