Oracle 高水位线和全表扫描(一)

2014-11-24 17:57:41 · 作者: · 浏览: 4

高水位线好比水库中储水的水位线,用于描述数据库中段的扩展方式。高水位线对全表扫描方式有着至关重要的影响。当使用delete 操作
表记录时,高水位线并不会下降,随之导致的是全表扫描的实际开销并没有任何减少。本文给出高水位线的描述,如何降低高水位线,以及高水


位线对全表扫描的影响。


一、何谓高水位线
如前所述,类似于水库中储水的水位线。只不过在数据库中用于描述段的扩展方式。
可以将数据段或索引段等想象为一个从左到右依次排开的一系列块。当这些块中未填充任何数据时,高水位线位于块的最左端(底端)
随着记录的不断增加,新块不断地被填充并使用,高水位线随之向右移动。高水位线之上为未格式化的数据块。
删除(delete)操作之后,高水位线之下的块处于空闲状态,但高水位线并不随之下降,直到重建,截断或收缩表段。
全表扫描会扫描高水位线之下的所有块,包括空闲数据块(执行了delete操作)。


二、演示高水位线与全表扫描


SQL> create table t -->创建测试表
2 as
3 select rownum as id,
4 round(dbms_random.normal*1000) AS val1,
5 dbms_random.string('p',250) AS pad
6 from dual
7 connect by level <=10000;


Table created.


SQL> exec dbms_stats.gather_table_stats('SCOTT','T',cascade=>true); -->收集统计信息


SQL> @Tab_Stat -->从dba_tab_statistics中获得表对象的统计信息,此时无empty_blocks的信息
Enter value for input_table_name: t
Enter value for input_owner: scott


NUM_ROWS BLKS EM_BLKS AVG_SPACE CHAIN_CNT AVG_ROW_LEN AVG_ROWS_PER_BLOCK LST_ANLY STA
---------- ---------- ---------- ---------- ---------- ----------- ------------------ --------- ---
10000 387 0 0 0 259 26 03-NOV-11 NO


/**************************************************/
/* Author: Robinson Cheng */
/* Blog: http://blog.csdn.net/robinson_0612 */
/* MSN: robinson_0612@hotmail.com */
/* QQ: 645746311 */
/**************************************************/


SQL> analyze table t compute statistics; -->执行analyze


SQL> @Tab_Stat -->此时的empty_blocks值为125
Enter value for input_table_name: t
Enter value for input_owner: scott


NUM_ROWS BLKS EM_BLKS AVG_SPACE CHAIN_CNT AVG_ROW_LEN AVG_ROWS_PER_BLOCK LST_ANLY STA
---------- ---------- ---------- ---------- ---------- ----------- ------------------ --------- ---
10000 387 125 920 0 262 26 03-NOV-11 NO


SQL> col segment_name format a15
SQL> select segment_name,segment_type,blocks,extents from dba_segments -->查看表段上的块的信息
2 where segment_name='T' and owner='SCOTT';


SEGMENT_NAME SEGMENT_TYPE BLOCKS EXTENTS -->此数据字典中记录的块数为512块(包含了已使用块与空闲块)
--------------- ------------------ ---------- ----------
T TABLE 512 19


SQL> set autotrace traceonly; -->开启autotrace
SQL> select count(*) from t; -->此时SQL语句的执行计划为全表扫描(执行计划中部分信息被省略)


Execution Plan
----------------------------------------------------------
Plan hash value: 2966233522


-------------------------------------------------------------------
| Id | Operation | Name | Rows | Cost (%CPU)| Time |
-------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 1 | 86 (0)| 00:00:02 |
| 1 | SORT AGGREGATE | | 1 | | |
| 2 | TABLE ACCESS FULL| T | 10000 | 86 (0)| 00:00:02 |
-------------------------------------------------------------------


Statistics
----------------------------------------------------------
1 recursive calls
0 db block gets
375 consistent gets -->consistent gets的值为375
0 physical reads


SQL> set autotrace off;
SQL> delete from t where rownum<=9900; -->删除大多数的记录,删除后剩余记录值为100


9900 rows deleted.
SQL> commit;


SQL> exec dbms_stats.gather_table_stats('SCOTT','T',cascade=>true); -->收集统计信息


SQL> analyze table t compute statistics; -->收集统计信息


SQL> @Tab_Stat -->此时对