dbms_xplan之display_cursor函数的使用

        DBMS_XPLAN包中display_cursor函数不同于display函数,display_cursor用于显示SQL语句的真实的执行计划,在大多数情况下,显示真实
的执行计划有助于更好的分析SQL语句的全过程,尤其是运行此SQL语句实时的I/O开销。通过对比预估的I/O与真实的I/O开销来判断SQL语句所存
在问题,如缺少统计信息,SQL语句执行的次数,根据实际中间结果集的大小来选择合适的连接方式等。本文仅仅讲述display_cursor函数的使
用。

        有关执行计划中各字段模块的描述请参考: 执行计划中各字段各模块描述
        有关由SQL语句来获取执行计划请参考:     使用 EXPLAIN PLAN 获取SQL语句执行计划
        有关使用autotrace来获取执行计划请参考: 启用 AUTOTRACE 功能
        有关dbms_xplan之display函数请参考:     dbms_xplan之display函数的使用

一、display_cursor函数用法
1、display_cursor函数语法

DBMS_XPLAN.DISPLAY_CURSOR(
 sql_id        IN  VARCHAR2  DEFAULT  NULL,
 child_number  IN  NUMBER    DEFAULT  NULL,
 format        IN  VARCHAR2  DEFAULT  'TYPICAL');

2、display_cursor函数参数描述
        sql_id
                指定位于库缓存执行计划中SQL语句的父游标。默认值为null。当使用默认值时当前会话的最后一条SQL语句的执行计划将被返回
                可以通过查询V$SQL 或 V$SQLAREA的SQL_ID列来获得SQL语句的SQL_ID。
        child_number
                指定父游标下子游标的序号。即指定被返回执行计划的SQL语句的子游标。默认值为0。如果为null,则sql_id所指父游标下所有子游标
                的执行计划都将被返回。
        format
                控制SQL语句执行计划的输出部分,即哪些可以显示哪些不显示。使用与display函数的format参数与修饰符在这里同样适用。
                除此之外当在开启statistics_level=all时或使用gather_plan_statistics提示可以获得执行计划中实时的统计信息
                有关详细的format格式描述请参考:dbms_xplan之display函数的使用 中format参数的描述

        下面给出启用统计信息时format新增的修饰符
                iostats   控制I/O统计的显示
                last      默认,显示所有执行计算过的统计。如果指定该值,则只显示最后一次执行的统计信息
                memstats  控制pga相关统计的显示
                allstats  此为iostats memstats的快捷方式,即allstats包含了iostats和memstats
                run_stats_last 等同于iostats last。只能用于oracle 10g R1
                run_stats_tot  等同于iostats。只能用于oracle 10g R1                  

二、演示使用display_cursor函数获取执行计划    
        1、当前数据库版本以及加载执行计划到库缓存               

SQL> select * from v$version where rownum<2;                      

BANNER
----------------------------------------------------------------
Oracle Database 10g Release 10.2.0.3.0 - 64bit Production         

SQL> SELECT ename,dname,loc
  2  FROM   emp e, dept d
  3  WHERE  e.deptno = d.deptno
  4  AND    e.empno  = 7788;                                      

ENAME      DNAME          LOC
---------- -------------- -------------
SCOTT      RESEARCH       DALLAS

        2、查看真实的执行计划              

/*----------------不传递任何参数给display_cursor函数,显示当前会话最后一条SQL语句的执行计划-------------*/
/**************************************************/
/* Author: Robinson Cheng                         */
/* Blog:   http://blog.csdn.net/robinson_0612     */
/* MSN:    robinson_0612@hotmail.com              */
/* QQ:     645746311                              */
/**************************************************/
SQL> select * from table(dbms_xplan.display_cursor(null,null));                                                       

PLAN_TABLE_OUTPUT
------------------------------------------------------------------------------------------
SQL_ID  a67wqmkfb9j65, child number 0
-------------------------------------
SELECT ename,dname,loc FROM   emp e, dept d WHERE  e.deptno = d.deptno AND
e.empno  = 7788                                                                                                       

Plan hash value: 2385808155                                                                                           

----------------------------------------------------------------------------------------
| Id  | Operation                    | Name    | Rows  | Bytes | Cost (%CPU)| Time     |
----------------------------------------------------------------------------------------
|   0 | SELECT STATEMENT             |         |       |       |     3 (100)|          |
|   1 |  NESTED LOOPS                |         |     1 |    63 |     3   (0)| 00:00:01 |
|   2 |   TABLE ACCESS BY INDEX ROWID| EMP     |     1 |    33 |     2   (0)| 00:00:01 |
|*  3 |    INDEX UNIQUE SCAN         | PK_EMP  |     1 |       |     1   (0)| 00:00:01 |
|   4 |   TABLE ACCESS BY INDEX ROWID| DEPT    |   409 | 12270 |     1   (0)| 00:00:01 |
|*  5 |    INDEX UNIQUE SCAN         | PK_DEPT |     1 |       |     0   (0)|          |
----------------------------------------------------------------------------------------                              

Predicate Information (identified by operation id):
---------------------------------------------------                                                                   

   3 - access("E"."EMPNO"=7788)
   5 - access("E"."DEPTNO"="D"."DEPTNO")                                                                              

/*------------------- 获得SQL语句的SQL_ID,可以看出此SQL_ID与上面显示的执行计划中的SQL_ID一致 ----------*/
SQL> select sql_id,address,plan_hash_value,hash_value,child_number from v$sql
  2  where sql_text like '%SELECT ename%' and sql_text not like '%from v$sql%';                                       

SQL_ID        ADDRESS          PLAN_HASH_VALUE HASH_VALUE CHILD_NUMBER
------------- ---------------- --------------- ---------- ------------
a67wqmkfb9j65 0000000091DBFBC8      2385808155 2629092549            0                                                

/*-------------- 传递SQL_ID以及format参数,并配合修饰符控制执行计划的输出 ------------------------*/
SQL> select * from table(dbms_xplan.display_cursor('a67wqmkfb9j65',null,'typical -predicate -rows'));                 

PLAN_TABLE_OUTPUT
------------------------------------------------------------------------------------
SQL_ID  a67wqmkfb9j65, child number 0
-------------------------------------
SELECT ename,dname,loc FROM   emp e, dept d WHERE  e.deptno = d.deptno
AND    e.empno  = 7788                                                                                                

Plan hash value: 2385808155                                                                                           

--------------------------------------------------------------------------------
| Id  | Operation                    | Name    | Bytes | Cost (%CPU)| Time     |
--------------------------------------------------------------------------------
|   0 | SELECT STATEMENT             |         |       |     3 (100)|          |
|   1 |  NESTED LOOPS                |         |    63 |     3   (0)| 00:00:01 |
|   2 |   TABLE ACCESS BY INDEX ROWID| EMP     |    33 |     2   (0)| 00:00:01 |
|   3 |    INDEX UNIQUE SCAN         | PK_EMP  |       |     1   (0)| 00:00:01 |
|   4 |   TABLE ACCESS BY INDEX ROWID| DEPT    | 12270 |     1   (0)| 00:00:01 |
|   5 |    INDEX UNIQUE SCAN         | PK_DEPT |       |     0   (0)|          |
--------------------------------------------------------------------------------

        3、查看真实执行计划并获得统计信息
                前提条件
                        设置参数statistics_level为all,可以基于session级别以及实例级别
                        或者启用gather_plan_statistics提示      

/*-------------查看实例参数statistics_level的值,并在会话级别将其设定为all  ---------*/
SQL> show parameter statistics_le                                                                                      

NAME                                 TYPE        VALUE
------------------------------------ ----------- ------------------------------
statistics_level                     string      ALL                                                                   

SQL> alter session set statistics_level=all;                                                                           

Session altered.                                                                                                       

SQL> select e.ename,e.sal,s.grade
  2  from emp e
  3  join salgrade s
  4  on e.sal between losal and hisal
  5  and e.deptno = 20;                                                                                                

ENAME             SAL      GRADE
---------- ---------- ----------
SCOTT            3000          4
FORD             3000          4
JONES            2975          4
ADAMS            1100          1
SMITH             800          1                                                                                       

/*------- 执行上述SQL语句后获得其真实的执行计划,使用了iostats last -predicate -note 修饰符控制显示输出 -----*/
SQL> set pagesize 0
SQL> select * from table(dbms_xplan.display_cursor(null,null,'iostats last -predicate -note'));
SQL_ID  243b0tpjxj6wv, child number 0
-------------------------------------
select e.ename,e.sal,s.grade from emp e join salgrade s on e.sal between losal and
hisal and e.deptno = 20                                                                                                

Plan hash value: 4204027666                                                                                            

-------------------------------------------------------------------------------------------
| Id  | Operation            | Name     | Starts | E-Rows | A-Rows |   A-Time   | Buffers |
-------------------------------------------------------------------------------------------
|   1 |  MERGE JOIN          |          |      1 |      1 |      5 |00:00:00.01 |      14 |
|   2 |   SORT JOIN          |          |      1 |      5 |      5 |00:00:00.01 |       7 |
|   3 |    TABLE ACCESS FULL | EMP      |      1 |      5 |      5 |00:00:00.01 |       7 |
|   4 |   FILTER             |          |      5 |        |      5 |00:00:00.01 |       7 |
|   5 |    SORT JOIN         |          |      5 |      5 |     14 |00:00:00.01 |       7 |
|   6 |     TABLE ACCESS FULL| SALGRADE |      1 |      5 |      5 |00:00:00.01 |       7 |
-------------------------------------------------------------------------------------------                            

/*---------------- 修改会话级别的参数statistics_level为typical并验证修改结果 ----------------*/
SQL> alter session set statistics_level=typical;                                                                       

SQL> col name format a40
SQL> col value format a25
SQL> col display_value format a25
SQL> select name, value, display_value, isses_modifiable
  2  from v$parameter
  3  where isses_modifiable = 'TRUE'
  4  and name like '%&input_name%';
Enter value for input_name: statistics_level
old   4: and name like '%&input_name%'
new   4: and name like '%statistics_level%'                                                                            

NAME                                     VALUE                     DISPLAY_VALUE             ISSES
---------------------------------------- ------------------------- ------------------------- -----
statistics_level                         TYPICAL                   TYPICAL                   TRUE                      

/*-------- 使用提示gather_plan_statistics,并获得其真实执行计划,使用了allstats -rows修饰符控制显示输出 ---*/
SQL> set pagesize 180
SQL> SELECT /*+ gather_plan_statistics */ ename,dname,loc
  2  FROM   emp e, dept d
  3  WHERE  e.deptno = d.deptno
  4  AND    d.deptno=20 ORDER BY 1,2,3;                                                                                

ENAME      DNAME          LOC
---------- -------------- -------------
ADAMS      RESEARCH       DALLAS
FORD       RESEARCH       DALLAS
JONES      RESEARCH       DALLAS
SCOTT      RESEARCH       DALLAS
SMITH      RESEARCH       DALLAS                                                                                       

SQL> select * from table(dbms_xplan.display_cursor(null,null,'allstats -rows'));                                       

PLAN_TABLE_OUTPUT
-------------------------------------------------------------------------------------------------
SQL_ID  d2hh42yzqqjz7, child number 0
-------------------------------------
SELECT /*+ gather_plan_statistics */ ename,dname,loc FROM   emp e, dept d WHERE  e.deptno = d.deptno AND
d.deptno=20 ORDER BY 1,2,3                                                                                             

Plan hash value: 3339094711                                                                                            

---------------------------------------------------------------------------------------------------------------------
| Id  | Operation                     | Name    | Starts | A-Rows |   A-Time   | Buffers |  OMem |  1Mem |  O/1/M   |
---------------------------------------------------------------------------------------------------------------------
|   1 |  SORT ORDER BY                |         |      1 |      5 |00:00:00.01 |       9 |  2048 |  2048 |     1/0/0|
|   2 |   NESTED LOOPS                |         |      1 |      5 |00:00:00.01 |       9 |       |       |          |
|   3 |    TABLE ACCESS BY INDEX ROWID| DEPT    |      1 |      1 |00:00:00.01 |       2 |       |       |          |
|*  4 |     INDEX UNIQUE SCAN         | PK_DEPT |      1 |      1 |00:00:00.01 |       1 |       |       |          |
|*  5 |    TABLE ACCESS FULL          | EMP     |      1 |      5 |00:00:00.01 |       7 |       |       |          |
---------------------------------------------------------------------------------------------------------------------  

Predicate Information (identified by operation id):
---------------------------------------------------                                                                    

   4 - access("D"."DEPTNO"=20)
   5 - filter("E"."DEPTNO"=20)                                                                                         

Note
-----
   - dynamic sampling used for this statement

三、总结
        1、与display函数不同,display_cursor显示的为真实的执行计划
        2、对于format参数,使用与display函数的各个值,同样适用于display_cursor函数
        3、当statistics_level为all或使用gather_plan_statistics提示可以获得执行时的统计信息
        4、根据真实与预估的统计信息可以初步判断SQL效率低下的原因,如统计信息的准确性、主要的开销位于那些步骤等              

时间: 2025-01-31 09:33:08

dbms_xplan之display_cursor函数的使用的相关文章

PL/SQL --&amp;gt; 动态SQL调用包中函数或过程

      动态SQL主要是用于针对不同的条件或查询任务来生成不同的SQL语句.最常用的方法是直接使用EXECUTE IMMEDIATE来执行动态SQL语句字符串或字符串变量.但是对于系统自定义的包或用户自定的包其下的函数或过程,不能等同于DDL以及DML的调用,其方式稍有差异.如下见本文的描述.       有关动态SQL的描述,请参考:           PL/SQL --> 动态SQL           PL/SQL --> 动态SQL的常见错误   1.动态SQL调用包中过程不正确

PL/SQL下SQL结果集以html形式发送邮件

       在运维的过程中,有时候需要定时将SQL查询的数据结果集以html表格形式发送邮件,因此需要将SQL查询得到的结果集拼接成html代码.对于这种情形通常有二种方式来完成.一是直接使用cron job来定时轮询并借助os级别的邮件程序来完成.其查询结果集可以直接在SQL*Plus下通过设置html标签自动实现html表格形式.一种方式是在Oracle中使用scheduler job来定时轮询.这种方式需要我们手动拼接html代码.本文即是对第二种情形展开描述. 关于PL/SQL下如何发

SQLplus 下行预取特性

   通常情况下数据库引擎每访问一个数据块将产生至少一个逻辑读.而行预取与逻辑读息息相关.行预取是指当客户端从数据库获取数据时可以采用单行也可以采用多行方式返回数据.当采用多行方式时,则会预取多条记录存储在客户端内存中以避免后续多次该数据的请求所致的各种开销(LIO,PIO,NET IO).一般预取行数越大,则所产生的开销越小,当达到临界值时其变化不大.一.演示 1.创建演示表 scott@CNMMBO> select * from v$version where rownum<2; BANN

VirtualBox 扩展包卸载或安装失败(VERR_ALREADY_EXISTS)

       最近在卸载VirtualBox出现了无法卸载的错误.提示为Failed to install the extension. The installer failed with exit code 1: VBoxExtPackHelperApp.exe: error: Failed to rename the extension pack directory: VERR_ALREADY_EXISTS.  而且也无法再次成功安装.本文是对此给出的描述及其解决办法.   1.故障描述  

ORA-27102: out of memory 故障

      最近的UAT数据库迁移,由于是多个DB需要迁移到同一台机器,一部分完成后,启动后续数据库碰到了ORA-27102错误,提示内存超出,查看系统可用内存,远大于需要启动数据库的sga和pga,究竟是怎么一回事呢?如果你也碰到类似错误,不妨往下看.   1.故障现象oracle@v2013db02u:~> cat /etc/issue Welcome to SUSE Linux Enterprise Server 10 SP3 (x86_64) - Kernel \r (\l). orac

执行计划中各字段各模块描述

      在SQL语句的执行计划中,包含很多字段项和很多模块,其不同字段代表了不同的含义且在不同的情形下某些字段.模块显示或不显示,下面的描述给出了执行计划中各字段的含义以及各模块的描述.        有关执行计划中各字段模块的描述请参考: 执行计划中各字段各模块描述        有关由SQL语句来获取执行计划请参考:     使用 EXPLAIN PLAN 获取SQL语句执行计划        有关使用autotrace来获取执行计划请参考:启用 AUTOTRACE 功能       有

ORA-31623: a job is not attached to this session via the specified handle

    在使用Oracel Datapump API时碰到ORA-31623(a job is not attached to this session via the specified handle)错误,从故障描述来看提示 job并没有成功附加到当前session指定的handle.该package的导入导出一直正常运行究竟是什么原因导致的呢?我们拭目以待...... 1.故障环境 SQL> select * from v$version where rownum<2; BANNER -

PL/SQL 游标变量

    游标变量与游标相似,有其共性,也有其不同点.就其共性来说两者都是指向多行查询的结果集中的当前行.都要经历声明,打开,检索与关闭的过程.所不同的是游标与游标变量类似于常量与变量.游标是静态的,而游标变量是动态的,因为游标变量并不与某个特定的查询相绑定.所以,游标变量可以打开任何类型兼容的查询.其次可以将游标变量作为参数传递给本地和存储子程序.本文主要描述游标变量的使用.   一.什么是游标变量    显示游标用于命名一个工作区域,其中保存多行查询的信息,而且该游标始终指向工作区域的内容.而

Oracle 11g RAC crs_stat 命令结果完整显示

Oracle 11g RAC中crs_stat命令较之前的版本多出了很多新的不同的资源类型,缺省情况下,使用crs_stat -t来查看资源是密密麻麻一大片,看起来着实费力.作者Paul Elbow, Enkitec为我们提供了一个crsstat脚本以更清晰的格式来展现Oracle 11g RAC下的所有资源类型,见本文下面的描述.   1.安装crsstat  通常情况下,建议使用root用户或者oracle用户来安装到 /usr/local/bin 目录  当脚本首次运行时,会在安装目录产生