14 Oracle Hunting Queries that ever Developer should know to Root Out Performance Issues

0
5495
Acclerate Oracle Queries by Performance Tuning

Summary of List of Oracle queries that every Analytics Developer should know.

Current Login Sessions
Where are my logins coming from
Currently executing SQL
Session Resource Consumption
Statements consuming the most resources
Statements performing full table scans
Displaying an execution plan
Index Usage
Hard Parsing Statistics
Queries Using Literal Values
Table Information
Column Information
Index Information
Stale Stats or Missing Stats

Contributors: David Berry, David Stoll.

Size matters and these hunting queries can provide insight to getting light weight and fast data sets when your SQL performance matters. It is detailed work but it is worth it to Pay the PRICE for PERFORMANCE

The hardest thing about performance is knowing what you need to measure, that must be tied back to your mission statement — Peter Drucker

— Current Login Sessions
SELECT
username, osuser, program, module,
machine, terminal, process,
to_char(logon_time, ‘YYYY-MM-DD HH24:MI:SS’)
AS logon_time,
status,
CASE status
WHEN ‘ACTIVE’ THEN NULL
ELSE last_call_et
END as idle_time
FROM v$session
WHERE type = ‘USER’;

— Where are my logins coming from
SELECT
username, osuser, program, module,
machine, process,
count(1) as login_count
FROM v$session
WHERE type = ‘USER’
GROUP BY
username, osuser, program, module,
machine, process;

— Currently executing SQL
SELECT
s.sid, s.username, s.osuser,
s.machine, s.process, s.program, s.module,
q.sql_text, q.optimizer_cost,
s.blocking_session, bs.username as blocking_user,
bs.machine as blocking_machine, bs.module as blocking_module,
bq.sql_text AS blocking_sql, s.event AS wait_event,
q.sql_fulltext
FROM v$session s
INNER JOIN v$sql q
ON s.sql_id = q.sql_id
LEFT OUTER JOIN v$session bs — blocking sessions
ON s.blocking_session = bs.sid
LEFT OUTER JOIN v$sql bq — blocking queries
ON bs.sql_id = bq.sql_id
WHERE s.type = ‘USER’;

— Session Resource Consumption
SELECT
s.sid, s.username, s.osuser,
to_char(sm.begin_time, ‘HH24:MI:ss’) AS interval_start,
to_char(sm.end_time, ‘HH24:MI:ss’) AS interval_end,
s.machine, s.process, s.program, s.module,
sm.cpu, sm.pga_memory, sm.logical_reads, sm.physical_reads,
sm.hard_parses, sm.soft_parses,
s.logon_time
FROM v$session s
INNER JOIN v$sessmetric sm
ON sm.session_id = s.sid
WHERE s.type = ‘USER’
ORDER BY sm.cpu DESC;

— Statements consuming the most resources
SELECT * FROM
(
SELECT sql_id, sql_text, executions,
elapsed_time, cpu_time, buffer_gets, disk_reads,
elapsed_time / executions AS avg_elapsed_time,
cpu_time / executions AS avg_cpu_time,
buffer_gets / executions as avg_buffer_gets,
disk_reads / executions as avg_disk_reads
FROM v$sqlstats
WHERE executions > 0
ORDER BY elapsed_time / executions DESC
)
WHERE rownum <= 25;

— Statements performing full table scans
SELECT
pl.object_owner, pl.object_name,
pl.sql_id, q.sql_text, q.module,
pl.operation, pl.options, pl.cost, pl.cpu_cost, pl.io_cost,
q.executions
FROM v$sql_plan pl
INNER JOIN v$sql q
ON pl.sql_id = q.sql_id
WHERE
(pl.operation = ‘TABLE ACCESS’ AND pl.options = ‘FULL’)
OR (pl.operation = ‘INDEX’ AND pl.options = ‘FAST FULL SCAN’)
OR (pl.operation = ‘INDEX’ AND pl.options = ‘FULL SCAN (MIN/MAX)’)
OR (pl.operation = ‘INDEX’ AND pl.options = ‘FULL SCAN’)
ORDER BY pl.object_owner, pl.object_name;

— Displaying an execution plan
SELECT plan_table_output
FROM
table(dbms_xplan.display_cursor(‘<>’,
null,’typical’));

— Index Usage
WITH index_usage AS
(
SELECT pl.sql_id, pl.object_owner, pl.object_name, pl.operation,
pl.options, count(1) as use_count
FROM v$sql_plan pl
WHERE pl.operation = ‘INDEX’
GROUP BY pl.sql_id, pl.object_owner, pl.object_name, pl.operation,
pl.options
)
SELECT
ix.table_owner, ix.table_name, ix.index_name, iu.operation,
iu.options, ss.sql_text, ss.executions
FROM all_indexes ix
LEFT OUTER JOIN index_usage iu
ON ix.owner = iu.object_owner
AND ix.index_name = iu.object_name
LEFT OUTER JOIN v$sqlstats ss
ON iu.sql_id = ss.sql_id
WHERE ix.owner = ‘<>’
ORDER BY ix.table_name, ix.index_name;

— Hard Parsing Statistics
SELECT sn.statistic#, sn.name, s.value
FROM v$sysstat s
INNER JOIN v$statname sn
ON s.statistic# = sn.statistic#
WHERE sn.name in (
‘parse time cpu’,
‘parse count (hard)’)

— Queries Using Literal Values
WITH statements AS
(
SELECT force_matching_signature,
count(1) OVER (PARTITION BY force_matching_signature) AS statement_count,
row_number() OVER (PARTITION BY force_matching_signature
ORDER BY last_load_time DESC) AS row_index,
parsing_schema_name,
sql_text
FROM v$sql
WHERE force_matching_signature > 0
AND force_matching_signature <> exact_matching_signature
)
SELECT
force_matching_signature, parsing_schema_name,
sql_text, statement_count
FROM statements
WHERE row_index = 1
AND statement_count >= 5;

— Table Information
SELECT
t.owner, t.table_name, t.num_rows, t.avg_row_len,
t.blocks AS blocks_below_hwm, t.empty_blocks,
s.blocks AS segment_blocks,
s.bytes / 1048576 AS size_in_mb,
to_char(t.last_analyzed, ‘YYYY-MM-DD HH24:MI’)
AS last_analyzed
FROM all_tables t
INNER JOIN dba_segments s
ON t.owner = s.OWNER AND t.table_name = s.segment_name
WHERE t.owner = ‘<>’;

— Column Information
SELECT column_name, avg_col_len,
num_distinct, num_nulls,
FROM all_tab_columns
WHERE table_name = ‘<>’
ORDER BY column_id DESC

— Index Information
SELECT ix.owner, ix.index_name, ix.table_name,
ix.distinct_keys, ix.leaf_blocks,
s.blocks AS segment_blocks,
s.bytes / 1048576 AS size_in_mb,
to_char(ix.last_analyzed, ‘YYYY-MM-DD HH24:MI’)
AS last_analyzed
FROM all_indexes ix
INNER JOIN dba_segments s
ON ix.owner = s.OWNER
AND ix.index_name = s.segment_name
ORDER BY ix.table_name, ix.index_name

— if statistics are stale (haven’t been refreshed in a long time).
SELECT table_name, sysdate, last_analyzed, sysdate – last_analyzed
FROM sys.all_tables
ORDER by 4 desc;