一、相关测试

(1)创建测试表

SQL> create table tt (id number);

Table created.

此时表没有分析,是原始的数据,即8个数据块。

SQL>SELECT segment_name,segment_type,blocks FROM dba_segments  WHERE segment_name='TT';

SEGMENT_NAME    SEGMENT_TYPE     BLOCKS

--------------- --------------- ----------

TT              TABLE                    8

SQL> SELECT table_name,num_rows,blocks,empty_blocks FROM user_tables  WHERE table_name='TT';

TABLE_NAME        NUM_ROWS     BLOCKS EMPTY_BLOCKS

--------------- ---------- ---------- ------------

TT


(2)向表中插入一些测试数据

SQL> declare

2  i number;

3  begin

4  for i in 110000 loop

5   insert into tt values(i);

6  end loop;

7  commit;

8  end;

9  /

PL/SQL procedure successfully completed.


(3)在次查看表的信息

SQL> SELECT table_name,num_rows,blocks,empty_blocks FROM user_tables  WHERE table_name='TT';

TABLE_NAME        NUM_ROWS     BLOCKS EMPTY_BLOCKS

--------------- ---------- ---------- ------------

TT

SQL>  SELECT segment_name,segment_type,blocks FROM dba_segments  WHERE segment_name='TT';

SEGMENT_NAME    SEGMENT_TYPE        BLOCKS

--------------- --------------- ----------

TT              TABLE                   24

此时表TT 占用的数据库已经是24个了。 但是user_tables 显示的信息还是为空。 因为没有做统计分析。


(4)收集统计信息

SQL> exec DBMS_STATS.GATHER_TABLE_STATS('SYS','TT');

PL/SQL procedure successfully completed.

SQL>  SELECT segment_name,segment_type,blocks FROM dba_segments  WHERE segment_name='TT';

SEGMENT_NAME    SEGMENT_TYPE        BLOCKS

--------------- --------------- ----------

TT              TABLE                   24

SQL> SELECT table_name,num_rows,blocks,empty_blocks FROM user_tables  WHERE table_name='TT';

TABLE_NAME        NUM_ROWS     BLOCKS EMPTY_BLOCKS

--------------- ---------- ---------- ------------

TT                   10000         20            0

此时user_tables 已经有了数据,显示的使用了20个数据块。 但是empty_blocks 还是为空。 这里要注意的地方。 这个字段只有使用analyze 收集统计信息之后才会有数据。


(5)使用analyze 收集统计信息

SQL> ANALYZE TABLE TT COMPUTE STATISTICS;

Table analyzed.

SQL> SELECT table_name,num_rows,blocks,empty_blocks FROM user_tables  WHERE table_name='TT';

TABLE_NAME        NUM_ROWS     BLOCKS EMPTY_BLOCKS

--------------- ---------- ---------- ------------

TT                   10000         20            3

-- 这里有显示空的数据库有3个。  注意:20+3=23. 比占用的24个数据块少一个。因为有一个数据库块被保留用作segment header。

SQL>  SELECT segment_name,segment_type,blocks FROM dba_segments  WHERE segment_name='TT';

SEGMENT_NAME    SEGMENT_TYPE        BLOCKS

--------------- --------------- ----------

TT              TABLE                   24


(6)delete 数据,不会降低高水位

SQL> delete from tt;

10000 rows deleted.

SQL> commit;

Commit complete.

SQL>  SELECT segment_name,segment_type,blocks FROM dba_segments  WHERE segment_name='TT';

SEGMENT_NAME    SEGMENT_TYPE        BLOCKS

--------------- --------------- ----------

TT              TABLE                   24

SQL> SELECT table_name,num_rows,blocks,empty_blocks FROM user_tables  WHERE table_name='TT';

TABLE_NAME        NUM_ROWS     BLOCKS EMPTY_BLOCKS

--------------- ---------- ---------- ------------

TT                   10000         20            3

SQL> analyze table tt compute statistics;

Table analyzed.

SQL> SELECT table_name,num_rows,blocks,empty_blocks FROM user_tables  WHERE table_name='TT';

TABLE_NAME        NUM_ROWS     BLOCKS EMPTY_BLOCKS

--------------- ---------- ---------- ------------

TT                       0         20            3

SQL>  SELECT segment_name,segment_type,blocks FROM dba_segments  WHERE segment_name='TT';

SEGMENT_NAME    SEGMENT_TYPE        BLOCKS

--------------- --------------- ----------

TT              TABLE                   24

SQL>


(7)truncate 表,可以降低高水位

SQL> truncate table tt;

Table truncated.

SQL>  SELECT segment_name,segment_type,blocks FROM dba_segments  WHERE segment_name='TT';

SEGMENT_NAME    SEGMENT_TYPE        BLOCKS

--------------- --------------- ----------

TT              TABLE                    8

SQL> SELECT table_name,num_rows,blocks,empty_blocks FROM user_tables  WHERE table_name='TT';

TABLE_NAME        NUM_ROWS     BLOCKS EMPTY_BLOCKS

--------------- ---------- ---------- ------------

TT                       0         20            3

-- 段的信息没有改变,收集一下统计信息看看

SQL> exec dbms_stats.gather_table_stats('SYS','TT');

PL/SQL procedure successfully completed.

SQL> SELECT table_name,num_rows,blocks,empty_blocks FROM user_tables  WHERE table_name='TT';

TABLE_NAME        NUM_ROWS     BLOCKS EMPTY_BLOCKS

--------------- ---------- ---------- ------------

TT                       0          0            3

SQL>  SELECT segment_name,segment_type,blocks FROM dba_segments  WHERE segment_name='TT';

SEGMENT_NAME    SEGMENT_TYPE        BLOCKS

--------------- --------------- ----------

TT              TABLE                    8

--段的信息已经改变,但是empty_blocks 段没有改变,该段只有使用analyze 才能改变。

SQL> analyze table tt compute statistics;

Table analyzed.

SQL> SELECT table_name,num_rows,blocks,empty_blocks FROM user_tables  WHERE table_name='TT';

TABLE_NAME        NUM_ROWS     BLOCKS EMPTY_BLOCKS

--------------- ---------- ---------- ------------

TT                       0          0            7

SQL>  SELECT segment_name,segment_type,blocks FROM dba_segments  WHERE segment_name='TT';

SEGMENT_NAME    SEGMENT_TYPE        BLOCKS

--------------- --------------- ----------

TT              TABLE                    8

SQL>

-- 总共8个数据块,7个为空,还有一个是segment header。


二、Alter table move 和Shrink 区别

1、Shrink

在10g之后,整理碎片消除行迁移的新增功能shrink space

SQL>alter table <table_name> shrink space [ <null> | compact | cascade ];

compact: 这个参数当系统的负载比较大时可以用,不降低HWM。如果系统负载较低时,直接用alter table table_name shrink space就一步到位了

cascade:这个参数是在shrink table的时候自动级联索引,相当于rebulid index。

以下SQL 基于普通表

shrink必须开启行迁移功能。

alter table table_name enable row movement ;

保持HWM,相当于把块中数据打结实了

alter table table_name shrink space compact;

回缩表与降低HWM

alter table table_name shrink space;

回缩表与相关索引,降低HWM

alter table table_name shrink space cascade;

回缩索引与降低HWM

alter index index_name shrink space

虽然在10g中可以用shrink ,但也有些限制:

1)对cluster,cluster table,或具有Long,lob类型列的对象 不起作用。

2)不支持具有function-based indexes 或 bitmap join indexes的表

3)不支持mapping 表或index-organized表。

4)不支持compressed 表


2、Move

通过desc table_name 来检查表中是否有LOB 字段, 如果表没有LOB字段,    直接 alter table move; 然后 rebuild index

如果表中包含了LOB字段,如用如下SQL:

SQL>alter table owner.table_name move tablespace tablespace_name lob (lob_column) store as lobsegment tablespace tablespace_name;

也可以单独move lob,但是表上的index 同样会失效。 所以在操作结束,需要对索引进行rebuild。

SQL>alter table owner.table_name move lob(lob_column) store as lobsegment tablespace tablespace_name ;

索引的rebuild:

首先用下面的SQL查看表上面有哪类索引:

SELECT a.owner,

a.index_name,

a.index_type,

a.partitioned,

a.status,

b.status p_status,

b.composite

FROM    dba_indexes a

LEFT JOIN

dba_ind_partitions b

ON a.owner = b.index_owner AND a.index_name = b.index_name

WHERE a.owner = '&owner' AND a.table_name = '&table_name';

对于普通索引直接rebuild online nologging parallel,

对于分区索引,必须单独rebuild 每个分区,

对于组合分区索引,必须单独rebuild 每个子分区。

Move 通过移动数据来来降低HWM,因此需要更多的磁盘空间。 Shrink 通过delete 和 insert, 会产生较多的undo 和redo。

shrink space收缩到数据存储的最小值,alter table move(不带参数)收缩到initial指定值,也可以用alter table test move storage(initial 500k)指定收缩的大小,这样可以达到shrink space效果。

总之,使用Move 效率会高点,但是会导致索引失效。Shrink 会产生undo 和redo,速度相对也慢一点。


oracle视频教程请关注:http://u.youku.com/user_video/id_UMzAzMjkxMjE2.html