在Oracle数据的存储中,可以把存储空间想象为一个水库,数据想象为水库中的水。水库中的水的位置有一条线叫做水位线,在Oracle中,这条线被称为高水位线(High-warter mark, HWM)。在数据库表刚建立的时候,由于没有任何数据,所以这个时候水位线是空的,也就是说HWM为最低值。当插入了数据以后,高水位线就会上涨,但是这里也有一个特性,就是如果你采用delete语句删除数据的话,数据虽然被删除了,但是高水位线却没有降低,还是你刚才删除数据以前那么高的水位。也就是说,这条高水位线在日常的增删操作中只会上涨,不会下跌。
下面我们来谈一下Oracle中Select语句的特性。Select语句会对表中的数据进行一次扫描,但是究竟扫描多少数据存储块呢,这个并不是说数据库中有多少数据,Oracle就扫描这么大的数据块,而是Oracle会扫描高水位线以下的数据块。现在来想象一下,如果刚才是一张刚刚建立的空表,你进行了一次Select操作,那么由于高水位线HWM在最低的0位置上,所以没有数据块需要被扫描,扫描时间会极短。而如果这个时候你首先插入了一千万条数据,然后再用delete语句删除这一千万条数据。由于插入了一千万条数据,所以这个时候的高水位线就在一千万条数据这里。后来删除这一千万条数据的时候,由于delete语句不影响高水位线,所以高水位线依然在一千万条数据这里。这个时候再一次用select语句进行扫描,虽然这个时候表中没有数据,但是由于扫描是按照高水位线来的,所以需要把一千万条数据的存储空间都要扫描一次,也就是说这次扫描所需要的时间和扫描一千万条数据所需要的时间是一样多的。所以有时候有人总是经常说,怎么我的表中没有几条数据,但是还是这么慢呢,这个时候其实奥秘就是这里的高水位线了。
那有没有办法让高水位线下降呢,其实有一种比较简单的方法,那就是采用TRUNCATE语句进行删除数据。采用TRUNCATE语句删除一个表的数据的时候,类似于重新建立了表,不仅把数据都删除了,还把HWM给清空恢复为0。所以如果需要把表清空,在有可能利用TRUNCATE语句来删除数据的时候就利用TRUNCATE语句来删除表,特别是那种数据量有可能很大的临时存储表。
在手动段空间管理(Manual Segment Space Management)中,段中只有一个HWM,但是在Oracle 9i Release1才添加的自动段空间管理(Automatic Segment Space Management)中,又有了一个低HWM的概念出来。为什么有了HWM还又有一个低HWM呢,这个是因为自动段空间管理的特性造成的。在手段段空间管理中,当数据插入以后,如果是插入到新的数据块中,数据块就会被自动格式化等待数据访问。而在自动段空间管理中,数据插入到新的数据块以后,数据块并没有被格式化,而是在第一次访问这个数据块的时候才格式化这个块。所以我们又需要一条水位线,用来标示已经被格式化的块。这条水位线就叫做低HWM。一般来说,低HWM肯定是低于等于HWM的。
下面的方法都可以降低高水位线标记。
1. 执行表重建指令 alter table table_name move;
在线转移表空间ALTER TABLE ... MOVE TABLESPACE ..
当你创建了一个对象如表以后,不管你有没有插入数据,它都会占用一些块,ORACLE也会给它分配必要的空间.同样,用ALTER TABLE MOVE释放自由空间后,还是保留了一些空间给这个表.
ALTER TABLE ... MOVE 后面不跟参数也行,不跟参数表还是在原来的表空间,Move后记住重建索引. 如果以后还要继续向这个表增加数据,没有必要move, 只是释放出来的空间,只能这个表用,其他的表或者segment无法使用该空间。
2. 执行alter table table_name shrink space;
注意,此命令为Oracle 10g新增功能,再执行该指令之前必须允许行移动 alter table table_name enable row movement;
3. 复制要保留的数据到临时表t,drop原表,然后rename临时表t为原表
4. 用逻辑导入导出: Emp/Imp
5. Alter table table_name deallocate unused
注:这证明,DEALLOCATE UNUSED为释放HWM上面的未使用空间,但是并不会释放HWM下面的自由空间,也不会移动HWM的位置.
6. 尽量使用truncate.
以上部分来自dave的博客
http://blog.csdn.net/tianlesoftware/archive/2009/10/21/4707900.aspx
关于HWM的试验
SQL> create table scott.aspen as select * from dba_objects;
表已创建。
SQL> analyze table scott.aspen estimate statistics;
表已分析。
SQL> select blocks,empty_blocks,num_rows
2 from dba_tables
3 where owner='SCOTT' and table_name='ASPEN';
BLOCKS EMPTY_BLOCKS NUM_ROWS
---------- ------------ ----------
1031 121 70062
SQL> delete from scott.aspen;
已删除68295行。
SQL> commit;
提交完成。
--删除表的数据后重新分析表
SQL> analyze table scott.aspen estimate statistics;
表已分析。
--得到表的HWM和delete前的是一样的,说明delete不会减小表的HWM
SQL> select blocks,empty_blocks,num_rows
2 from dba_tables
3 where owner='SCOTT' and table_name='ASPEN';
BLOCKS EMPTY_BLOCKS NUM_ROWS
---------- ------------ ----------
1031 121 0
--如果是用truncate又会是什么结果呢
SQL> insert into scott.aspen select * from dba_objects;
已创建68295行。
SQL> commit;
提交完成。
SQL> analyze table scott.aspen estimate statistics;
表已分析。
SQL> select blocks,empty_blocks,num_rows
2 from dba_tables
3 where owner='SCOTT' and table_name='ASPEN';
BLOCKS EMPTY_BLOCKS NUM_ROWS
---------- ------------ ----------
1031 121 70062
SQL> truncate table scott.aspen;
表被截断。
Truncate表之后重新分析表
SQL> analyze table scott.aspen estimate statistics;
表已分析。
-- 结果:truncate则会将hwm置为0
SQL> select blocks,empty_blocks,num_rows
2 from dba_tables
3 where owner='SCOTT' and table_name='ASPEN';
BLOCKS EMPTY_BLOCKS NUM_ROWS
---------- ------------ ----------
0 8 0
手动修改表的HWM
SQL> truncate table scott.aspen;
表被截断。
SQL> insert into scott.aspen select * from dba_objects;
已创建68295行。
SQL> commit;
提交完成。
SQL> analyze table scott.aspen estimate statistics;
表已分析。
SQL> select blocks,empty_blocks
2 from dba_tables
3 where owner='SCOTT' and table_name='ASPEN';
BLOCKS EMPTY_BLOCKS
---------- ------------
1126 26
SQL> delete from scott.aspen;
已删除68295行。
SQL> commit;
提交完成。
SQL> analyze table scott.aspen estimate statistics;
表已分析。
--上面的试验已经证明delete不会改小表的HWM,所以这里查询出来的值或delete之前的值是一样一样的
SQL> select blocks,empty_blocks
2 from dba_tables
3 where owner='SCOTT' and table_name='ASPEN';
BLOCKS EMPTY_BLOCKS
---------- ------------
1126 26
SQL> alter table scott.aspen enable row movement;
表已更改。
SQL> alter table scott.aspen shrink space;
表已更改。
SQL> analyze table scott.aspen estimate statistics;
表已分析。
--手动修改表的HWM后的结果
SQL> select blocks,empty_blocks
2 from dba_tables
3 where owner='SCOTT' and table_name='ASPEN';
BLOCKS EMPTY_BLOCKS
---------- ------------
1 7
HWM在插入数据时,当现有空间不足而进行空间的扩展时会向上移.
SQL> analyze table scott.aspen estimate statistics;
表已分析。
SQL> select blocks,empty_blocks,num_rows
2 from dba_tables
3 where owner='SCOTT' and table_name='ASPEN';
BLOCKS EMPTY_BLOCKS NUM_ROWS
---------- ------------ ----------
1 7 2
SQL> insert into scott.aspen select * from dba_objects;
已创建68295行。
SQL> commit;
提交完成。
SQL> analyze table scott.aspen estimate statistics;
表已分析。
SQL> select blocks,empty_blocks,num_rows
2 from dba_tables
3 where owner='SCOTT' and table_name='ASPEN';
BLOCKS EMPTY_BLOCKS NUM_ROWS
---------- ------------ ----------
1126 26 75909
当用直接路径插入行时,即使HWM以下有空闲的数据库块,键入在插入数据时使用了append关键字,
则在插入时使用HWM以上的数据块,此时HWM会自动增大
SQL> delete from scott.aspen;
已删除68297行。
SQL> commit;
提交完成。
SQL> select count(*) from scott.aspen;
COUNT(*)
----------
0
SQL> analyze table scott.aspen estimate statistics;
表已分析。
SQL> select blocks,empty_blocks,num_rows
2 from dba_tables
3 where owner='SCOTT' and table_name='ASPEN';
BLOCKS EMPTY_BLOCKS NUM_ROWS
---------- ------------ ----------
1126 26 0
SQL> insert into scott.aspen select * from dba_objects;
已创建68295行。
SQL> commit;
提交完成。
SQL> analyze table scott.aspen estimate statistics;
表已分析。
SQL> select blocks,empty_blocks,num_rows
2 from dba_tables
3 where owner='SCOTT' and table_name='ASPEN';
BLOCKS EMPTY_BLOCKS NUM_ROWS
---------- ------------ ----------
1126 26 68103
--从上面可以看出使用insert时 在HWM以下有足够空闲的数据库块情况下,HWM的值不会增加
SQL> truncate table scott.aspen;
表被截断。
SQL> insert into scott.aspen select * from dba_objects;
已创建68295行。
SQL> commit;
提交完成。
SQL> analyze table scott.aspen estimate statistics;
表已分析。
SQL> select blocks,empty_blocks,num_rows
2 from dba_tables
3 where owner='SCOTT' and table_name='ASPEN';
BLOCKS EMPTY_BLOCKS NUM_ROWS
---------- ------------ ----------
1126 26 75906
SQL> delete from scott.aspen;
已删除68295行。
SQL> commit;
提交完成。
SQL> analyze table scott.aspen estimate statistics;
表已分析。
SQL> select blocks,empty_blocks,num_rows
2 from dba_tables
3 where owner='SCOTT' and table_name='ASPEN';
BLOCKS EMPTY_BLOCKS NUM_ROWS
---------- ------------ ----------
1126 26 0
SQL> insert /*+append*/ into scott.aspen select * from dba_objects;
已创建68295行。
SQL> commit;
提交完成。
SQL> analyze table scott.aspen estimate statistics;
表已分析。
SQL> select blocks,empty_blocks,num_rows
2 from dba_tables
3 where owner='SCOTT' and table_name='ASPEN';
BLOCKS EMPTY_BLOCKS NUM_ROWS
---------- ------------ ----------
2173 3 65155
--从上面可以看出使用/*+append*/ insert时 即使在HWM以下有足够空闲的数据库块,HWM的值也会增加