如何查找SQL执行的真实执行计划

有时候我们会发现,一个看似同样的SQL语句,在PL/SQL Developer里面运行速度很快,放到应用程序里面运行却很糟糕。
为什么会这样呢?原来,同样的SQL在不同环境下,执行计划是会发生改变的,那么,如何得到最准确的执行计划呢?

oracle 10g的DBMS_XPLAN包中display_cursor函数不同于display函数,display_cursor用于显示SQL语句的真实的执行计划,在大多数情况下,
显示真实的执行计划有助于更好的分析SQL语句的全过程,尤其是运行此SQL语句实时的I/O开销。通过对比预估的I/O与真实的I/O开销来判断
SQL语句所存在问题,如缺少统计信息,SQL语句执行的次数,根据实际中间结果集的大小来选择合适的连接方式等。本文仅仅讲述
display_cursor函数的使用。
 
一、display_cursor函数用法
 1、display_cursor函数语法
 
 DBMS_XPLAN.DISPLAY_CURSOR(                        
  sql_id        IN  VARCHAR2  DEFAULT  NULL,       
  cursor_child_no  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。
         cursor_child_no
                 指定父游标下子游标的序号。即指定被返回执行计划的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                 

抓一个最近一小时最消耗IO的SQL:
SELECT sql_id, COUNT(*)
  FROM gv$active_session_history ash, gv$event_name evt
 WHERE ash.sample_time > SYSDATE - 1 / 24
   AND ash.session_state = 'WAITING'
   AND ash.event_id = evt.event_id
   AND evt.wait_class = 'User I/O'
 GROUP BY sql_id
 ORDER BY COUNT(*) DESC;

执行上面的SQL:
 SQL> SELECT sql_id, COUNT(*)
  FROM gv$active_session_history ash, gv$event_name evt
  2    3   WHERE ash.sample_time > SYSDATE - 1 / 24
  4     AND ash.session_state = 'WAITING'
  5     AND ash.event_id = evt.event_id
  6     AND evt.wait_class = 'User I/O'
  7   GROUP BY sql_id
  8   ORDER BY COUNT(*) DESC;

SQL_ID          COUNT(*)
------------- ----------
g7fu6qba82m6b        668
63r47zyphdk06        526
9f5m4wd88nc1h        514
593p47drw5fhk        232
br91w16jzy4fu        120
4fvwyjpnh6tp7         78
gm0nrbfuj8kzr         70
2184k363hw4xd         68
gc4dajs7g5myy         46
8vrk9sfuwfdgq         42
ccpnb4dwdmq21         40

查看SQL的执行计划:
SELECT * FROM TABLE(dbms_xplan.display_cursor('g7fu6qba82m6b'));  

在SQLPLUS中执行:
SQL> set pagesize 2000
SQL> SELECT * FROM TABLE(dbms_xplan.display_cursor('g7fu6qba82m6b'));

PLAN_TABLE_OUTPUT
------------------------------------------------------------------------------------------------------------------------
SQL_ID  g7fu6qba82m6b, child number 0
-------------------------------------
UPDATE "CPDDS_PDATA"."CDM_LEDGER" SET "CSTM_NAME" = :a1,"CSTM_NO" =
:a2,"PAPER_TYPE" = :a3,"PAPER_NO" = :a4,"CURR_TYPE" = :a5,"SVT_NO" =
:a6,"BAL_DIR" = :a7,"BAL" = :a8,"AVAL_BAL" = :a9,"NORM_FRATIO" =
:a10,"PK_BAL" = :a11,"DR_ACCU" = :a12,"CR_ACCU" = :a13,"LAST_TRAN_DATE" =
:a14,"LAST_TRAN_TIME" = :a15,"PRT_LINE_NUM" = :a16,"NOREG_PK_REC_NUM" =
:a17,"PK_NO" = :a18,"PWD" = :a19,"FLAG" = :a20,"FRZ_FLAG" =
:a21,"CARD_HOLD_FLAG" = :a22,"PK_HOLD_FLAG" = :a23,"BGN_INT_DATE" =
:a24,"OPEN_DATE" = :a25,"ACC_HOLD_FLAG" = :a26,"CLS_DATE" =
:a27,"OPEN_TLR" = :a28,"CLS_TLR" = :a29,"CLS_INT" = :a30,"OPEN_INST" =
:a31,"ADD_NUM" = :a32,"DAC" = :a33,"FRZ_TIMES1" = :a34,"FRZ_TIMES2" =
:a35,"HOST_SEQNO" = :a36,"D_UPDATE_DATE" = :a37 WHERE "ACC" = :b0

Plan hash value: 319441092

-----------------------------------------------------------------------------------
| Id  | Operation          | Name         | Rows  | Bytes | Cost (%CPU)| Time     |
-----------------------------------------------------------------------------------
|   0 | UPDATE STATEMENT   |              |       |       |     3 (100)|          |
|   1 |  UPDATE            | CDM_LEDGER   |       |       |            |          |
|*  2 |   INDEX UNIQUE SCAN| I_CDM_LEDGER |     1 |   269 |     2   (0)| 00:00:01 |
-----------------------------------------------------------------------------------

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

   2 - access("ACC"=:B0)


29 rows selected.


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

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

lordcoohoo

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值