1. 准备知识:ORACLE的逻辑存储管理.
1.2 区:由一系列相邻的块而组成,这也是ORACLE空间分配的基本单位,举个例子来说,当我们创建一个表PM_USER时,首先ORACLE会分配一区的空间给这个表,随着不断的INSERT数据到PM_USER,原来的这个区容不下插入的数据时,ORACLE是以区为单位进行扩展的,也就是说再分配多少个区给PM_USER,而不是多少个块. 1.3 段:是由一系列的区所组成,一般来说,当创建一个对象时(表,索引),就会分配一个段给这个对象.所以从某种意义上来说,段就是某种特定的数据.如CREATE TABLE PM_USER,这个段就是数据段,而CREATE INDEX ON PM_USER(NAME),ORACLE同样会分配一个段给这个索引,但这是一个索引段了.查询段的信息可以通过数据字典: SELECT * FROM USER_SEGMENTS来获得, 1.4 表空间:包含段,区及块.表空间的数据物理上储存在其所在的数据文件中.一个数据库至少要有一个表空间.
( INITIAL 64K MINEXTENTS 1 MAXEXTENTS UNLIMITED );
如何知道一个表的HWM? a) 首先对表进行分析: ANALYZE TABLE <tablename> ESTIMATE/COMPUTE STATISTICS; b) 查看相关信息: SELECT blocks, empty_blocks, num_rows FROM user_tables WHERE table_name = <tablename>;
EMPTY_BLOCKS 代表分配给该表,但是在水线以上的数据库块,即从来没有使用的数据块. 示例进行说明: SQL>SELECT segment_name,segment_type,blocks FROM dba_segments WHERE segment_name='TEST'; SEGMENT_NA SEGMENT_TYPE BLOCKS ---------- ------------------ ---------- TEST TABLE 8 TEST TABLE 8 SQL> ANALYZE TABLE TEST COMPUTE STATISTICS; 表已分析。 SQL> SELECT table_name,num_rows,blocks,empty_blocks FROM user_tables WHERE table_name='TEST'; TABLE_NAME NUM_ROWS BLOCKS EMPTY_BLOCKS ------------------------ -------------------- ---------- ------------ TEST 10 1 6
SQL> SELECT COUNT (DISTINCT DBMS_ROWID.ROWID_BLOCK_NUMBER(rowid) || 2 DBMS_ROWID.ROWID_RELATIVE_FNO(rowid)) "Used"FROM TEST; Used ---------- 1 SQL> DELETE from TEST; 已删除10行。 SQL> COMMIT; 提交完成。 SQL> ANALYZE TABLE TEST ESTIMATE STATISTICS; 表已分析。 SQL> SELECT table_name,num_rows,blocks,empty_blocks FROM user_tables WHERE table_name='TEST'; TABLE_NAME NUM_ROWS BLOCKS EMPTY_BLOCKS ------------------ ------------ --- ---------- - -------------------- TEST 0 1 6 SQL> SELECT COUNT (DISTINCT DBMS_ROWID.ROWID_BLOCK_NUMBER(rowid) || 2 DBMS_ROWID.ROWID_RELATIVE_FNO(rowid)) "Used" FROM TEST;
---------- 0 注:Used = 0 这表名没有任何数据库块容纳数据,即表中无数据 SQL> TRUNCATE TABLE TEST; 表被截断。 SQL> ANALYZE TABLE TEST COMPUTE STATISTICS; 表已分析。 SQL> SELECT table_name,num_rows,blocks,empty_blocks FROM user_tables WHERE table_name='TEST'; TABLE_NAME NUM_ROWS BLOCKS EMPTY_BLOCKS ---------------------------------------- ---------- ------------ TEST 0 0 7
WHERE segment_name='TEST'; SEGMENT_NA SEGMENT_TYPE BLOCKS -------------------- ------------------------- ------------- TEST TABLE 8 TEST TABLE 8
3. Oracle表段中的高水位线HWM
在ORACLE中,执行对表的删除操作不会降低该表的高水位线。而全表扫描将始终读取一个段(extent)中所有低于高水位线标记的块。如果在执行删除操作后不降低高水位线标记,则将导致查询语句的性能低下。
在线转移表空间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.
在9I中: 1. 如果是INEXTENT, 可以使ALTER TABLE TABLENAME DEALLOCATE UNUSED将HWM以上所有没使用的空间释放 2. 如果MINEXTENT >HWM 则释放MINEXTENTS 以上的空间。如果要释放HWM以上的空间则使用KEEP 0。 ALTER TABLE TABLESNAME DEALLOCATE UNUSED KEEP 0; 3. TRUNCATE TABLE DROP STORAGE(缺省值)命令可以将MINEXTENT 之上的空间完全释放(交还给操作系统),并且重置HWM。 4. 如果仅是要移动HWM,而不想让表长时间锁住,可以用TRUNCATE TABLE REUSE STORAGE,仅将HWM重置。 5. ALTER TABLE MOVE会将HWM移动,但在MOVE时需要双倍的表空间,而且如果表上有索引的话,需要重构索引 6. DELETE表不会重置HWM,也不会释放自由的空间(也就是说DELETE空出来的空间只能给对象本身将来的INSERT/UPDATE使用,不能给其它的对象使用)
1. 可以使用ALTER TABLE TEST_TAB SHRINK SPACE命令来联机移动HWM, 2. 如果要同时压缩表的索引,可以发布:ALTER TABLE TEST_TAB SHRINK SPACE CASCADE 5. HWM 特点:
这就好比是水库的水位,当涨水时,水位往上移,当水退出后,最高水位的痕迹还是清淅可见. ORACLE 不会释放空间以供其他对象使用,有一条简单的理由:由于空间是为新插入的行保留的,并且要适应现有行的增长。被占用的最高空间称为最高使用标记 (HWM),
HWM本身的信息是储存在段头.在段空间是手工管理方式时,ORACLE是通过FREELIST(一个单向链表)来管理段内的空间分配.在段空间是自动管理方式时(ASSM),ORACLE是通过BITMAP来管理段内的空间分配. 4. ORACLE的全表扫描是读取高水位标记(HWM)以下的所有块. 所以问题就产生了.当用户发出一个全表扫描时,ORACLE 始终必须从段一直扫描到 HWM,即使它什么也没有发现。该任务延长了全表扫描的时间。
例如,通过直接加载插入(用 APPEND 提示插入)或通过 SQL*LOADER 直接路径 数据块直接置于 HWM 之上。它下面的空间就浪费掉了。 |
|
来自: guolijiegg > 《oracle》