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
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
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
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
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
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
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