dbms_stats 导入导出表统计信息(二)

2014-11-24 14:29:52 · 作者: · 浏览: 2
---------
Plan hash value: 453826725
------------------------------------------------------------------------------------
| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |
------------------------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 1 | 6 | 58 (0)| 00:00:01 |
| 1 | SORT GROUP BY NOSORT | | 1 | 6 | 58 (0)| 00:00:01 |
|* 2 | INDEX FAST FULL SCAN| I_T1_OWNER | 31349 | 183K| 58 (0)| 00:00:01 |
------------------------------------------------------------------------------------
--上面的执行计划中使用了最新的统计信息,而且预估的行数31349接近于表上的行数
3、导入过旧的统计信息并对比执行计划
[sql]
--下面使用import_table_stats导入之前过旧的统计信息
scott@USBO> exec dbms_stats.import_table_stats(ownname => 'SCOTT', tabname => 'T1', stattab => 'ST_T1', -
> statid => 'A', no_invalidate => true);
PL/SQL procedure successfully completed.
--再次查看原SQL的执行计划
scott@USBO> select owner,count(*) from t1 where owner='SYS' group by owner;
Execution Plan
----------------------------------------------------------
Plan hash value: 832695366
-----------------------------------------------------------------------------------
| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |
-----------------------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 1 | 8 | 4 (0)| 00:00:01 |
| 1 | SORT GROUP BY NOSORT| | 1 | 8 | 4 (0)| 00:00:01 |
|* 2 | INDEX RANGE SCAN | I_T1_OWNER | 1425 | 11400 | 4 (0)| 00:00:01 |
-----------------------------------------------------------------------------------
--Author : Leshami Blog : http://blog.csdn.net/leshami
--从上面的执行计划中来看,尽管执行计划与先前的两个执行计划相同,但预估的行数是之前的行数,只有1425条记录
--也即是由于过时的统计信息造成的
4、批量导出表统计信息
[sql]
--下面的匿名pl/sql块可以批量导出统计信息,可以用于SQL语句涉及到多表的情形,可以把相关的表统计信息全部导出
--需要注意的是表的名字不要超过28,因为我这里定义的统计信息备份表以"S_"开头占据了2个字符
--可以根据自己的情形修改其代码,如添加表空间参数等。
--对于披量导入表统计信息的脚本,大家可以参照下面的脚本修改,使用过程import_table_stats
DECLARE
v_table_name VARCHAR2 (30);
v_stat_name VARCHAR2 (35);
v_sql_stat VARCHAR2 (200);
v_schema VARCHAR2 (30) := 'SCOTT';
--Define your table you want to export stat
CURSOR cur_tab
IS
SELECT table_name
FROM dba_tables
WHERE table_name IN ('EMP', 'DEPT', 'BONUS');
BEGIN
FOR cur_rec IN cur_tab
LOOP
v_stat_name := 'S_' || cur_rec.table_name;
v_sql_stat := 'BEGIN DBMS_STATS.create_stat_table (''' || v_schema || ''' , ''' || v_stat_name || '''); END;';
-- DBMS_OUTPUT.put_line (v_sql_stat);
EXECUTE IMMEDIATE v_sql_stat;
v_sql_stat := 'BEGIN DBMS_STATS.export_table_stats(''' || v_schema || ''',tabname=>''' || cur_rec.table_name || ''',stattab=>''' || v_stat_name || '''); END;';
-- DBMS_OUTPUT.put_line (v_sql_stat);
EXECUTE IMMEDIATE v_sql_stat;
END LOOP;
END;