1.使用DDL操作可以让ORACLE再次执行目标SQL时使用硬解析,但其影响范围太广,因为一旦对某个表执行了DDL操作,再次执行与这个表相关的所有SQL时就会全部使用硬解析。
这是很不好的,特别是对于OLTP类型的应用系统而言,因为这可能会导致短时间内硬解析数量剧增,进而影响系统的性能。
2.可以使用DBMS_SHARED_POOL.PURGE来删除指定的缓存在库缓存中的Shared Cursor,该方式影响范围仅限于该目标SQL所对应的Shared Cursor,
也就是说它可以做到只让Oracle在执行目标SQL时使用硬解析,在执行所有其他SQL时都和原来一样保持不变。
SQL> SELECT SQL_TEXT,SQL_ID,ADDRESS,HASH_VALUE FROM v$SQLAREA WHERE SQL_TEXT LIKE 'SELECT COUNT(*) FROM T1%';
SQL> EXEC SYS.DBMS_SHARED_POOL.PURGE('address,hash_value','c') --其中这里第二个输入参数是常量c,表示要删除的是Shared Cursor。
3.Child Cursor中除了会存储目标SQL的解析树和执行计划之外,还会存储该SQL所使用的绑定变量的类型和长度,这意味着即使该SQL的SQL文本没有发生任何改变,
只要其SQL文本中文本型绑定变量的定义发生了改变,那么该SQL再次执行时可能还是做了硬解析(参考3.2.5 绑定变量分级)
explain plan for +目标SQL
SELECT * FROM TABLE(DBMS_XPLAN.DISPLAY)
补充:SELECT DBMS_METADATA.GET_DDL('TABLE','PLAN_TABLES$','SYS') FROM DUAL --查看原代码
SELECT SID FROM v$MYSTAT WHERE ROWNUM<2; --查看当前SESSION的SESSION ID
A.用于在SQLPLUS中查看刚刚执行过的SQL的执行计划
SQL> SELECT * FROM TABLE(DBMS_XPLAN.DISPLAY_CURSOR(NULL,NULL,'ADVANCED'))
B.用于查看指定SQL的执行计划(只要目标SQL的执行计划所在的Child Cursor还没有被age out出Shared Pool,就可以用这种方法来查看该SQL的执行计划)
SQL>SELECT SQL_TEXT,SQL_ID,HASH_VALUE,CHILD_NUMBER FROM v$SQL WHERE SQL_TEXT LIKE 'SELECT EMPNO,ENAME%';
/* 只要目标SQL所对应的Child Cursor还在Library Cache中,我们就可以从v$SQL中查到目标SQL的Child Cursor的详细信息,包括SQL ID,SQL HASH VALUE,Child Cursor Number等
SQL> SELECT * FROM TABLE(DBMS_XPLAN.DISPLAY_CURSOR('sql_id/hash_value',child_cursor_number,'ADVANCED')) ;
案例:SELECT * FROM TABLE(DBMS_XPLAN.DISPLAY_CURSOR('3yfu3wh150aqt','0','ADVANCED')) ;
C.用于查看指定SQL的所有历史执行计划。
使用方法A、B能够显示目标SQL执行计划的前提条件是该SQL的执行计划还在Shared Pool中,而如果该SQL的执行计划已经被age out出Shared Pool,
那么只有该SQL的执行计划被ORACLE采集到AWR Repository中,我们就可以使用方法C来查看该SQL的所有历史执行计划。
SQL> SELECT * FROM TABLE(DBMS_XPLAN.DISPLAY_AWR('79glp919t7x4u')) ;
案例:模拟实验环境
1.手工收集一下AWR报告,采集完成后清空Shared Pool
SQL> EXEC DBMS_WORKLOAD_REPOSITORY.CREATE_SNAPSHOT();
SQL> ALTER SYSTEM FLUSH SHARED_POOL; --请勿随意在生产环境执行此语句
2.查看目标SQL的执行计划是否被age out出Shared Pool了
SQL> SELECT SQL_TEXT,SQL_ID,VERSION_COUNT,EXECUTIONS FROM v$SQLAREA WHERE SQL_TEXT LIKE 'SELECT COUNT(*) FROM T1%'; --此时已没有记录了
3.由于之前我们已经通过手工采集AWR报告的方式将目标SQL的执行计划采集到了AWR Repository中,所以现在我们可以通过执行方法C来得到该SQL的所有历史执行计划:
SQL> SELECT * FROM TABLE(DBMS_XPLAN.DISPLAY_AWR('79glp919t7x4u')) ;
SQL> set autot ?
Usage: SET AUTOT[RACE] {OFF | ON | TRACE[ONLY]} [EXP[LAIN]] [STAT[ISTICS]]
SQL>SET AUTOTRACE OFF (set autot off) 不显示 只有查询结果
SET AUTOTRACE ON (set autot on) 启动 : 查询结果 执行计划 统计信息
SET AUTOTRACE TRACEONLY(set autot trace) :不显示查询结果 只显示:执行结果数量 执行计划 统计信息
SET AUTOTRACE TRACEONLY EXPLAN (set autot trace exp) ;;:只显示:执行计划
SET AUTOTRACE TRACEONLY STATISTICS(set autot trace stat) ; 只显示 :执行结果数量 统计信息
缺陷:1.必须要等到语句真正执行完毕后,才可以出结果
2.无法看到表被访问了多少次。
SQL> alter session set statistics_level=all;
SQL> 执行语句
SQL> select * from table(dbms_xplan.display_cursor(null,null,'allstats last'));
优点:1.可以看出SQL语句对应的等待事件。
2.如果SQL语句中有函数调用,SQL中有SQL,将会都被列出,无所遁形。
3.可以方便的看出处理的行数,产生的逻辑读。
4.可以方便的看出解析时间和执行时间。
5.可以跟踪整个程序包。
缺陷:1.无法判断表被访问了多少次。
2.步骤繁琐。
3.执行计划中的谓词部分不能清晰的展示出来。
A.先根据AWR报告找到目标SQL
生成AWR报告
SQL>@?\rdbms\admin\awrrpt.sql
B.手工执行脚本$ORACLE_HOME/rdbms/admin/awrsqrpt.sql,并依次输入报告类型、要查看的快照范围、目标SQL ID和所有生成的AWR SQL报告的名称;
SQL> @?\rdbms\admin\awrsqrpt.sql';(建议的快照范围和AWR报告的范围一致)
--------------------------------额外补充--------------------------------------------------
创建plustrace角色,然后将角色赋给特定用户。Oracle已经提供了角色plustrace的创建脚本
@/u01/app/oracle/product/11.2.0/dbhome_1/sqlplus/admin/plustrce.sql
把角色赋权给zw用户
SQL> grant plustrace to zw;
SQL> conn zw/zw
Connected.
SQL> set autotrace traceonly;
把角色赋予给所有用户
SQL> grant plustrace to public;