Skip to Main Content

Breadcrumb

Question and Answer

Tom Kyte

Thanks for the question, Robert.

Asked: April 19, 2007 - 6:27 pm UTC

Last updated: January 30, 2013 - 1:37 pm UTC

Version: 9.2.0

Viewed 10K+ times! This question is

You Asked

Tom,

Do you have a script that will format v$sql_plan into a good formatted output.
e.g. pass in the v$sql_plan.address and it will output a well formatted explain/access plan?

Thanks!

Robertt.

and Tom said...

Here is a quote from "Effective Oracle by Design" that shows one approach

<quote>
DBMS_XPLAN and V$SQL_PLAN

In Oracle9i Release 2 there is a new supplied package ¿ DBMS_XPLAN. If you edit the script utlxpls.sql, you'll discover in Oracle9i it is effectively one line long:

select plan_table_output 
from table( dbms_xplan.display( 'plan_table',null,'serial'))


If you edit that same script in Oracle9i Release 1 or before, you'll find a huge query. DBMS_XPLAN.DISPLAY is a better method for querying and displaying the plan output. It is a function that simply returns a collection, which is a procedurally formatted explain plan output ¿ including the supplemental information at the bottom of the report ¿ which I have already pointed out, is new in Oracle9i Release2. It is a side effect of using this new package. So, if you do not have access to the utlxpls.sql script ¿ that simple query will perform the same function. In fact, that package is so good at adjusting its output based on the inputs ¿ you need not even supply the inputs as utlxpls.sql does, a simple:

Select * from TABLE(dbms_xplan.display)


suffices.

One of the nice features of this ¿ coupled with the V$SQL_PLAN dynamic performance view ¿ is that it becomes very easy to "dump" the query plans for already executed statements, right out of the database. Above, I have demonstrated how we can use an INSERT into the PLAN_TABLE and then run utlxpls or utlxplp to see the plan. Well, in 9iR2 using DBMS_XPLAN and a view we can create ¿ it becomes even easier. If you use a schema that has been granted SELECT on SYS.V_$SQL_PLAN directly, we'll be able to create this view:

ops$tkyte@ORA920> create or replace view dynamic_plan_table
  2  as
  3  select
  4   rawtohex(address) || '_' || child_number statement_id,
  5   sysdate timestamp, operation, options, object_node,
  6   object_owner, object_name, 0 object_instance,
  7   optimizer,  search_columns, id, parent_id, position,
  8   cost, cardinality, bytes, other_tag, partition_start,
  9   partition_stop, partition_id, other, distribution,
 10   cpu_cost, io_cost, temp_space, access_predicates,
 11   filter_predicates
 12   from v$sql_plan;

View created.


Now, we can query any plan out of the database with a single query:

ops$tkyte@ORA920> select plan_table_output
  2    from TABLE( dbms_xplan.display
  3                ( 'dynamic_plan_table',
  4                  (select rawtohex(address)||'_'||child_number x
  5                     from v$sql
  6  where sql_text='select * from t t1 where object_id > 32000' ),
  7                  'serial' ) )
  8  /

PLAN_TABLE_OUTPUT
------------------------------------------------------------------
| Id  | Operation                   | Name|Rows| Bytes |Cst(%CPU)|
------------------------------------------------------------------
|   0 | SELECT STATEMENT            |     |    |       |         |
|   1 |  TABLE ACCESS BY INDEX ROWID| T   |291 | 27936 | 25   (0)|
|*  2 |   INDEX RANGE SCAN          | T_PK|291 |       |  2   (0)|
------------------------------------------------------------------
Predicate Information (identified by operation id):
---------------------------------------------------

   2 - access("OBJECT_ID">32000)

13 rows selected.


The text in italics above, is a query that gets the STATEMENT_ID, which we are interested in. In this query, you can use whatever query/values you want which would identify the exact query plan you wanted to review.
</quote>

Rating

  (8 ratings)

Is this answer out of date? If it is, please let us know via a Comment

Comments

BINGO!

Robert Wood, April 23, 2007 - 1:47 pm UTC

Tom,

EXACTLY what I am looking for!

When I first got your "Effective Oracle by Design" we were not running 9i on our main database so I sort of skipped over the 9i specific stuff. While reviewing your book again over the weekend I saw this very thing you reference.
Thank you for the confimation of this technique by answering my question!

Thanks!

Robert.

SQL attempt

K, November 21, 2007 - 10:56 pm UTC

Unfortunately I'm at a site where I dont have any create privileges and am limited to:

SELECT ANY DICTIONARY
CREATE SESSION

so I couldn't construct the view. As a quick try I created an SQL script SQLPLAN_LIST.sql:

column "Operation" format A40
column "Object Name" format A30
column "PStart" format A6
column "PStop" format A6
column "I" format A3
column ACCESS_PREDICATES format A80
column FILTER_PREDICATES format A80
set lines 9999
set pages 99
set trimspool on
set verify off
set heading on

select  /*+ NO_MERGE */ rownum  || decode(access_predicates,NULL,decode(filter_predicates,NULL,'','*'),'*') "I",
        substr(lpad(' ',2*(depth-1)) || operation,1,20) || 
        decode(options,NULL,'',' (' || options || ')') "Operation",
        substr(object_name,1,30) "Object Name",
        cardinality "# Rows",
        bytes,
        cost,
        partition_start "PStart",
        partition_stop "PStop"
  from (select * from v$sql_plan where hash_value = &&1) a
  start with id = 0
  connect by prior id = parent_id
      and prior nvl(hash_value, 0 ) = nvl(hash_value, 0 )
  order by id, position
/
set heading off
select 'Access Predicates .....' from dual;
select * from (
select  /*+ NO_MERGE */ rownum  || decode(access_predicates,NULL,decode(filter_predicates,NULL,'','*'),'*') "I",
    decode(access_predicates,NULL,'###',access_predicates) "ACCESS_PREDICATES"
  from (select * from v$sql_plan where hash_value = &&1) a
  start with id = 0
  connect by prior id = parent_id
      and prior nvl(hash_value, 0 ) = nvl(hash_value, 0 )
  order by id, position
) where access_predicates <> '###'
/
set heading off
select 'Filter predicates .....' from dual;
select * from (
select  /*+ NO_MERGE */ rownum  || decode(access_predicates,NULL,decode(filter_predicates,NULL,'','*'),'*') "I",
    decode(filter_predicates,NULL,'###',filter_predicates) "FILTER_PREDICATES"
  from (select * from v$sql_plan where hash_value = &&1) a
  start with id = 0
  connect by prior id = parent_id
      and prior nvl(hash_value, 0 ) = nvl(hash_value, 0 )
  order by id, position
) where filter_predicates <> '###'
/


and then in use : 

SQL> @sqlplan_list 2942443644

1   SELECT STATEMENT                                                                                      60
2   SORT (UNIQUE)                                                                    6       1038         60
3     UNION-ALL
4       NESTED LOOPS                                                                 1        179         22
5         NESTED LOOPS                                                               1        163         21
6*          TABLE ACCESS (FULL)              DBT_C_I_REPLY_Q                         1        133         20
7           TABLE ACCESS (BY INDEX ROWID)    DBT_C_I_CONT_MSG                        1         30          1
8*            INDEX (UNIQUE SCAN)            DBO_C_I_CONT_MSG_01                     1
9*        TABLE ACCESS (BY INDEX ROWID)      DBT_C_I_CTL_RQST                        1         16          1
10*         INDEX (UNIQUE SCAN)              DBO_C_I_CTL_RQST_01                     1

Access Predicates .....

8*  "CONT"."BRCH_CODE"=:B0 AND "CONT"."SEQ_ID"="REPLY"."SEQ_ID"
10* "RQST"."BRCH_CODE"=:B0 AND "RQST"."RQST_ID"="CONT"."RQST_ID"

Filter predicates .....

6*  "REPLY"."BRCH_CODE"=:B0 AND ("REPLY"."STAT_CODE"=:B1 OR "REPLY"."STAT_CODE"=:B2)
9*  NVL("RQST"."STAT_CODE",'XX')<>'01' AND NVL("RQST"."STAT_CODE",'XX')<>'07'

The above had some lines deleted to make it more readable.

Provided here just in case it comes in handy for someone else in a similar position

Tom Kyte
November 26, 2007 - 10:28 am UTC

... so I couldn't construct the view. ...

you can always use WITH or an inline view instead.


how to use use/inline view in this case?

jim, February 17, 2011 - 2:11 am UTC

with dynamic_plan_table
as
( select
rawtohex(address) || '_' || child_number statement_id,
sysdate timestamp, operation, options, object_node,
object_owner, object_name, 0 object_instance,
optimizer, search_columns, id, parent_id, position,
cost, cardinality, bytes, other_tag, partition_start,
partition_stop, partition_id, other, distribution,
cpu_cost, io_cost, temp_space, access_predicates,
filter_predicates
from v$sql_plan)
select plan_table_output
from TABLE( dbms_xplan.display
( 'dynamic_plan_table',
(select rawtohex(address)||'_'||child_number x
from v$sql
where sql_text='select * from t t1 where object_id > 32000' ),
'serial' ) );


This gives me an error saying

ERROR at line 14:
ORA-32035: unreferenced query name defined in WITH clause


Tom Kyte
February 17, 2011 - 8:15 am UTC

it doesn't work that way - dbms_xplan cannot "see" your dynamic plan table - you would get "table or view does not exist"

remember, dbms_xplan is a bit of code that will run it's own SQL - it will try to query dynamic plan table - but will not be able to "find it"

How to write it then?

jim, February 17, 2011 - 10:40 am UTC

Thanks, Could you please tell me how to write the query then if I dont have permission to create view?

Thanks
Tom Kyte
February 17, 2011 - 12:10 pm UTC

In 10g and above, there is a dbms_xplan.display_cursor routine that you can use to display the information from v$sql_plan for any query in the shared pool.

if you have 10g and above, you can use that - you don't need the view.

http://docs.oracle.com/docs/cd/E11882_01/appdev.112/e16760/d_xplan.htm#ARPLS70136

Bala Baskaran, September 07, 2011 - 2:11 pm UTC

Hi Tom,

Greetigns!. Is there a way we could use dbms_xplan without creating this view, we are constrained by the oracle 9i environment.

The intent is to validate the plan on production to the development environment are the same / nearly the same.

Please enlighten us.

Thanks & Regards,
Bala

Tom Kyte
September 08, 2011 - 5:07 pm UTC

why is 9i constraining your to not use a view? I'm confused?


You could just compare plan hashes from v$sql_plan to see if plans are different, be a tad more efficient.

I don't know what a "nearly the same" plan is - a plan is either the same, or it isn't. There are not any "it is nearly the same" plans that I know of.

TW

A reader, November 11, 2011 - 8:24 am UTC

"The text in italics above, is a query that gets the STATEMENT_ID"

This is no text in italics above. Please don't waste my time
Tom Kyte
November 11, 2011 - 8:44 am UTC

it was a quote from a book - and the formatting didn't get copied over. Big deal, deal with it. If you cannot really figure it out, let me know and I'll point it out to you.

Don't waste my time.

v$sql_plan

Girish, January 24, 2013 - 10:25 pm UTC

Hi Tom,

Above response gave rise to a doubt.Is there a difference bwteen plan generated using trace file(I mean in tkprof) and exection plan found in v$sql_plan ?

Thanks,
Girish
Tom Kyte
January 30, 2013 - 1:37 pm UTC

the plan in the trace file - the row source plan - is the plan that was used by that sql query at that time.

the planS in v$sql represent the plans for the sql statement that are currently in the shared pool.

meaning - if you have a trace file from yesterday with a set of STAT records that have the plan used - you might find that that plan *isn't* in v$sql_plan right now because it got aged out.


so, the row source plan in the trace file, that is what was used when that trace file was generated.

the information in v$sql_plan - that is the set of plans being used for that query right now, this moment, and might be different than what you see in the trace file.

9i db performance

JO, May 21, 2013 - 11:51 am UTC

We had been assigned a task of Performance Statistics gathering in Production DBs.
For the same we followed the steps mentioned below

create temp table to hold the sql stats and plan namely sql_stats_temp,
and another table temp_tab with data FROM v$sql_plan,sql_stats_temp and joined it using rawtohex(address) & child_number of v$sql_plan = rawtohex(address) of sql_stats_temp .
alter sql_stats_temp,to add clob column for storing sql plan
and then wrote a script will fetch the sql stats and plan and load into sql_stats_temp table.
select plan_table_output
from TABLE( dbms_xplan.display( 'TEMP22_TABLE',rec1.sql_id,'serial')
)
).

But We get the No more data to read from socket error.

This is because of the size allocated to access_predicates,filter_predicates which is 4000.


--drop temp table
drop table sql_stats_temp;
drop table TEMP22_TABLE;

-- create temp table to hold the sql stats and plan
create table sql_stats_temp
as
SELECT *
FROM ( select v.Hash_value,
v.sql_text,
v.elapsed_time,
v.cpu_time,
v.ROWS_PROCESSED,
v.disk_reads,
v.buffer_gets,
v.executions,
v.first_load_time,
v.last_load_time,
v.address,
v.child_number
from v$sql v
where upper(v.sql_text) NOT like '%SYS%'
and upper(v.sql_text) NOT like'%DBMS_%'
and v.sql_text NOT like'%$%'
and upper(v.sql_text) NOT like'%SESSION%'
and upper(v.sql_text) NOT like'%TEMP22%'
and upper(v.sql_text) NOT like'%SQL_STATS_TEMP%'
and upper(v.sql_text) NOT like'%JAVA%'

-- where rawtohex(v.address)||'_'||child_number = t.statement_ID
order by elapsed_time DESC
)
WHERE rownum <= 1000;


CREATE table TEMP22_TABLE (STATEMENT_ID, TIMESTAMP, OPERATION, OPTIONS, OBJECT_NODE, OBJECT_OWNER, OBJECT_NAME, OBJECT_INSTANCE, OPTIMIZER, SEARCH_COLUMNS, ID, PARENT_ID, POSITION, COST, CARDINALITY, BYTES, OTHER_TAG, PARTITION_START, PARTITION_STOP, PARTITION_ID, OTHER, DISTRIBUTION, CPU_COST, IO_COST, TEMP_SPACE, ACCESS_PREDICATES, FILTER_PREDICATES)
AS
SELECT rawtohex(vp.address)
|| '_'
|| vp.child_number statement_id,
sysdate TIMESTAMP,
vp.operation,
vp.OPTIONS,
vp.object_node,
vp.object_owner,
vp.object_name,
0 object_instance,
vp.optimizer,
vp.search_columns,
vp.id,
vp.parent_id,
vp.position,
vp.cost,
vp.cardinality,
vp.bytes,
vp.other_tag,
vp.partition_start,
vp.partition_stop,
vp.partition_id,
vp.other,
vp.distribution,
vp.cpu_cost,
vp.io_cost,
vp.temp_space,
vp.access_predicates,
vp.filter_predicates
FROM v$sql_plan vp,
sql_stats_temp st
WHERE rawtohex(vp.address)
|| '_'
|| vp.child_number = rawtohex(st.address)
|| '_'
|| st.child_number;


--alter table to add clob column for storing sql plan
ALTER TABLE sql_stats_temp
ADD (sqlplan CLOB)

/

select * from V$sql where address='000000039B1CB0A0'


-- The below script will fetch the sql stats and plan and load into sql_stats_temp table.
Declare
v_plan CLOB;
Begin
FOR rec1 IN (select rawtohex(address)||'_'||child_number sql_id from sql_stats_temp) LOOP

v_plan:= NULL;

-- The below loop will get the plan of the sql statement and update the sqlplan clob column
FOR rec2 IN (

select plan_table_output
from TABLE( dbms_xplan.display( 'TEMP22_TABLE',rec1.sql_id,'serial')
)
) LOOP

v_plan:= v_plan||' '||rec2.plan_table_output;


END LOOP;
update sql_stats_temp
set sqlplan = v_plan
where rawtohex(address)||'_'||child_number = rec1.sql_id;
COMMIT;
END LOOP;
END;
/
--select * from sql_stats_temp

More to Explore

PL/SQL demos

Check out more PL/SQL tutorials on our LiveSQL tool.

PL/SQL docs

PL/SQL reference manual from the Oracle documentation library