,QIRUPDWLRQ 7HFKQRORJ\
DNDGLD
Oracle Datenbank Tuning
Author: Martin Zahn
Copyright © 2000 Akadia AG
All rights reserved
Akadia AG
Information Technology
Arvenweg 4
CH-3604 Thun
Tel
033 335 86 20
Fax
033 335 86 25
E-Mail
[email protected]
Web
http://www.akadia.ch
Inhaltsverzeichnis
1.
Der Tuning Prozess................................................................. 1
Grundsätzliches
2.
SQL-Optimierung..................................................................... 3
Grundregeln
Aufgaben des Optimizers
2.1 Optimizer Modes ........................................................................................... 4
Rule Based Optimizer
COST based Optimizer
2.2 Zugriffsmethoden .......................................................................................... 7
Full Table Scan
Index Scan
2.3 Indexierungs Strategie ................................................................................ 13
Regeln
Selektivität
B*TREE Index Analyse
2.4 Datenverteilung ........................................................................................... 15
2.5 Join Algorithmen ......................................................................................... 16
Sort-Merge Join
Hash-Join
Nested Loop
Star Joins
Beurteilung
2.6 Execution Plan ............................................................................................ 20
AutoTrace in SQL*Plus
TopSession im Enterprise Manager
TKPROF
2.7 Optimierung................................................................................................. 26
Hints
2.8 Regeln für gutes SQL.................................................................................. 28
Allgemeines
Outer Joins
DISTINCT
NOT IN
Inline Functions
3.
Tuning ausgewählter SQL-Statements................................ 30
3.1 COST- oder RULE-based Optimizer ? ........................................................ 30
COST-based
RULE-based
Index-Zugriff
3.2 Join-Methoden (Algorithmen) ...................................................................... 32
Join-Methoden
3.3 Ein oder mehrere Attribute in der Selektion ................................................ 36
SELECT * ...
3.4 Subquery Optimierung ................................................................................ 38
Variante 1
Variante 2
Variante 3
Variante 4
3.5 Zählen von Rows ........................................................................................ 40
COUNT
3.6 SQL-Funktionen die Indexzugriff verhindern............................................... 41
4.
Datenbank Tuning ................................................................ 42
Übersicht
4.1 Oracle Performance Pack ...........................................................................42
Overview Window
Buffer Cache Hit %
Library Cache Hit %
Data Dictionary Cache Hit %
Memory Sort Hit %
Rollback Nowait %
Rollback Segment Waits
Rollback Segments Shrinks
File I/O Rate
System I/O Rate
Throughput
Active Users
Circuit
Dispatcher
MTS
File I/O Rate Details
Free List Hit%
Latch
Library Cache Details
Lock
Memory Allocated
Network Bytes Rate
Network I/O Rate
Session Events
Parse Ratio
Processes
Queue
Redo Alloc Hit%
Redo Statistics
Session
Sort Rows Rate
Sqlarea
System Statistics
Table Access
Users logged on
Users running
Users waiting
Users waiting for Locks
Object Status
4.2 Detaillierte Analyse des Shared Pools ........................................................65
Memory Fragmentierung
Vorhandene Memory Segmente
Grosse Objecte lokalisieren
Grosse Anonymous PL/SQL Blocks
Grösse des Shared Pools bestimmen
Grösse des Reserved Pools bestimmen
Überwachung des Shared Pool
Output der Überwachung
5.
Locking Verhalten.................................................................. 81
5.1 Row Locks ...................................................................................................81
Row Share Table Locks (RS)
Row Exclusive Table Locks (RX)
5.2 Table Locks .................................................................................................82
5.3 Lock Manager Lock Types ..........................................................................82
$NDGLD $*
Oracle Datenbank Tuning
1.
Der Tuning Prozess
Grundsätzliches
Es gibt wenig allgemein gültige Rezepte, da beispielsweise ein Tuning für
ein Transaktionssystem (OLTP) anders vorgenommen werden muss als
ein Tuning für ein DataWareHouse (DSS).
Ein «seriöses» Tuning muss sich über einen grösseren Zeitraum erstrekken, Momentaufnahmen bilden die Mosaiksteine zum gesamten TuningProzess.
Tuning beginnt grundsätzlich bereits beim Design und begleitet
anschliessend die Applikation auf ihrem gesamten Lifecycle.
Ein Tuning beinhaltet immer mehrere Komponenten, die nur in ihrem
Gesamtzusammenhang den angestrebten Nutzen bringen.
Datenmodell
Korrekte
Indexierung
SQL-Optimierung
Tools
DB-Tuning
Physische
Implementierung
Betriebssystem
Optimierung
Datenmodell
• Keine oder übertriebene Normalisierung.
• Denormalisierung gezielt einsetzen.
• Gewährleistung der Integrität (Jede Tabelle hat einen Primary Key,
referentielle Integritätskontrolle durch Foreign Keys, UNIQUE Attribute,
CHECK Constraints, Datenbank-Trigger, etc).
• Nachführen von aufwendigen Berechnungen und Summenbildungen
(vermeiden von SELECT COUNT(*) für Statistiken).
Korrekte Indexierung
• Indexe nur dort wo sie «wirklich» einen Nutzen bringen.
• Vermeidung von Locking-Problemen bei gleichzeitigem Mehrfachzugriff auf Master-Detail Tabellen.
• Bitmapped Indexe für wenig selektive Attribute verwenden.
• Wenn nötig Index-Histogramme erstellen.
Ausgabe 5. Januar 1995
Seite 1Seite
von 82
-1
Oracle Datenbank Tuning
Tools
$NDGLD $*
• Möglichst moderne Tools einsetzen, die das Tuning unterstützen.
Datenmodell:
Designer/2000, S-Designer
Physische Implementation:
Designer/2000, PowerBuilder
Client / Server Tools:
PowerBuilder, Delphi, Visual C++, etc
Netzwerk:
Möglichst direkt über SQL*Net ohne
Umweg über ODBC.
Stored Procedures:
SQL-Navigator
• Nicht am Tool «vorbei programmieren», das heisst, dass man die Eigenschaften des Tools nutzen sollte und nicht einen Lösungsweg anstrebt, für den das Tool nicht geeignet ist.
Physische
Implementierung
• Tablespace Organisation, meist sind Index- und Tabellendaten getrennt.
• I/O-Tuning und Verteilung der Datenbank-Files auf mehrere Disks. Dabei ist Rücksicht zu nehmen auf RAID-Systeme. Grundsätzlich empfiehlt Oracle RAID 0 und RAID 1, aber nicht RAID 5.
• RAW-Devices einsetzen
(möglichst nicht mit normalen Filesystem mischen).
• Speicherstrukturen wie SGA, Shared Pool optimieren (DB_BUFFER,
SHARED_POOL_SIZE, SHARED_POOL_RESERVED_SIZE,
SHARED_POOL_RESERVED_MIN_ALLOC).
• Indexe überwachen und gegebenfalls neu erstellen.
DB-Tuning
• INIT.ORA Parameter optimieren
• Behebung von Engpässen aufgrund gleichzeitiger Beanspruchung der
Datenbank durch viele Prozesse (Free Lists, Rollback Segmente, Latches etc).
• Trennung von DSS und OLTP Datenbanken mittels Replikation.
SQL-Optimierung
• Richtiger Einsatz des Optimizers. Der COST-based Optimizer verlangt
Statistiken, die aufgebaut werden müssen.
• Index-Scan oder Full-Table Zugriff ?
• Optimizer Hints einsetzen.
• Alternativen für ein bestimmtes SQL-Statement suchen, testen.
• Transaktionskontrolle ist Sache des Programmierers. Transaktionen so
kurz wie möglich halten, Exceptions, Savepoints etc einbauen.
• Kein explizites Locking programmieren, SELECT FOR UPDATE nur wo
dies wirklich begründbar ist. Möglichst nie ein exclusives Table Locking
einsetzen. Grundsätzlich sollte das Lockingverhalten Oracle überlassen werden.
Betriebssystem
Optimierung
• Oracle verlangt genügend Memory für die gesamte SGA. Die SGA
MUSS unbedingt im Memory Platz finden. (Swapping, Paging der SGA
muss unbedingt verhindert werden).
• Auslastung überwachen (TOP, SAR, etc).
Die Bedeutung des INIT.ORA Tunings wird oft überschätzt, den grössten
Nutzen bezüglich Performance liegt in einem guten Design sowie guter
SQL-Programmierung. Messungen zur Performance immer dann vornehmen, wenn die Instance und Datenbank bereits lange im Betrieb waren.
Ein DB-Tuning nach dem Startup der DB ist unbrauchbar. Änderungen,
die nichts gebracht haben, wieder rückgängig machen. Unbedingt ein
Step-by-Step Vorgehen wählen und die Resultate der einzelnen Schritte
vergleichen.
Seite
2 von
Ausgabe
5.82
Januar
1995
Seite -2
$NDGLD $*
Oracle Datenbank Tuning
2.
SQL-Optimierung
Grundregeln
• Nur 1 - 2 % der SQL Befehle dürfen Performance Probleme aufweisen,
ansonsten muss in einem anderen Bereich gesucht werden (Datenmodell, Hardware, physische Implementation).
• Restriktionen frühzeitig erkennen, dies erfolgt immer in der WHERE
Klausel. Der EQUI-Join verhindert nur das Kartesische Produkt.
• Regeln einhalten, um entweder Fulltable-Scan oder Index-Zugriff zu
ermöglichen.
• Die Antwortzeit ist direkt von der Anzahl Rows des Resultats und nicht
von der Anzahl Rows in der Tabelle abhängig.
• Interaktive Abfragen sollten immer nur eine kleine Menge an Treffern
haben. Dies ist besonders wichtig im Client / Server Bereich.
• Bei Batchverarbeitung ist die gesamte Antwortzeit entscheidend, die
Zeit bis der erste Record gefunden wird ist unwichtig.
Aufgaben des
Optimizers
Der Optimizer versucht über eine frühzeitige Restriktion die ResultatsMenge so klein wie möglich zu halten.
• Auf Tabellen Ebene geschieht eine Einschränkung nur dann, wenn
nicht mehr die gesamte Tabelle gelesen werden muss. Dazu braucht es
eine WHERE-Klausel, welche auf indexierte Attribute appliziert werden
kann.
• Bei Joins kann über die Join-Attribute nur das Kartesische Produkt verhindert werden. Join-Attribute gelten deshalb nicht als Einschränkung
auf Tabellen Ebene.
• Die Aufgabe des Optimizer besteht also darin, aufgrund der WHERE
Klausel die kleinste Menge an Treffern zu finden, welche nicht für einen
Join verwendet werden. Diese gefundene Menge wird nun mit der Menge an Treffern aus der anderen Tabelle gejoint.
Um diese Aufgabe so effizient wie möglich zu erfüllen, erstellt der Optimizer einen Execution Plan, der kontrolliert und beeinflusst werden kann.
Als Grundregel kann gesagt werden, dass selektive Abfragen mit einem
kleinen Result-Set über einen Index-Scan gelesen werden sollten. Wird
dagegen die gesamte Tabelle gelesen (keine WHERE-Klausel) so ist ein
Fulltable-Scan deutlich effizienter. Wo die Grenze liegt, kann nicht genau
spezifiziert werden, da auch die Selektivität der Daten eine wichtige Rolle
spielt. Als ersten Ansatz kann von einer Annahme von
15 % ausgegangen werden.
Ausgabe 5. Januar 1995
Seite 3Seite
von 82
-3
$NDGLD $*
Oracle Datenbank Tuning
2.1
Optimizer Modes
Oracle 7 bietet zwei verschiedene Optimizer an:
• Rule Based Optimizer (RBO)
• Cost Based Optimizer (CBO)
Rule Based
Optimizer
Der Rule Based Optimizer entscheidet aufgrund des zur Verfügung stehenden Access Path und des zugehörigen Ranks nach einem fix
vorgegeben Schemas.
Vorteile des Rule Based Optimizers
• Bei Migrationen älterer Applikationen (vorallem V6 -> V7)
• Immer dann, wenn ein Fulltable-Scan möglichst verhindert werden soll.
• Der Parse Overhead ist sehr gering.
Rank Access Path
1
Single Row by ROWID
Beispiel: empno ist der Primary Key
SELECT
FROM
WHERE
FOR
rowid INTO var_rowid
emp
empno = 4711
UPDATE;
UPDATE emp SET ename = ’MILLER’
WHERE rowid = var_rowid;
2
3
4
Single Row by Cluster Join
Single Row by Hash Cluster Key with Unique or Primary Key
Single Row by Unique or Primary Key
Beispiel: empno ist der Primary Key
SELECT *
FROM emp
WHERE empno = 7900
5
6
7
8
Clustered Join
Hash Cluster Key
Index Cluster Key
Composite Index
Beispiel: Composite Index auf (job, deptno)
SELECT
FROM
WHERE
AND
9
*
emp
job = ’CLERK’
depno = 30;
Single Column Indexes
Beispiel: Index auf job
SELECT *
FROM emp
WHERE job = ’CLERK’;
10
Bounded Range Search on Indexed Columns
Beispiel: column ist an 1. Stelle in einem Composite Index
WHERE
WHERE
WHERE
WHERE
Seite
4 von
Ausgabe
5.82
Januar
1995
column
column
column
column
= expr
>[=] expr AND column <[=] expr
BETWEEN expr AND expr
LIKE ’ABC%’
Seite -4
$NDGLD $*
Oracle Datenbank Tuning
Rank Access Path
11
Unounded Range Search on Indexed Columns
Beispiel: column ist an 1. Stelle in einem Composite Index
WHERE column >[=] expr
WHERE column <[=] expr
12
13
Sort-Merge Join
MAX or MIN of Indexed Columns
Beispiel: Auf sal ist ein Index
SELECT MAX(sal) FROM emp;
14
ORDER BY on Indexed Columns
Damit dieser Path benutzt wird müssen folgende Bedingungen erfüllt sein:
1. Die Column der ORDER BY Cluase muss indexiert sein
2. Die Column muss UNIQUE sein.
3. NLS_SORT muss auf BINARY gestellt sein
Beispiel: empno ist der Primary Key
SELECT * FROM emp ORDER BY empno;
14
Full Table Scan
Der Rule Based Optimizer versucht also mit allen Mitteln einen FulltableScan zu verhindern. Oft wird jedoch die gesamte Tabelle gelesen, wo ein
Fulltable-Scan die effizienteste Methode darstellt.
Beachte
Bei Performance Problemen kann zuerst der andere Optimizer zugezogen werden, anschliessend werden die Execution Plans miteinander
verglichen.
Übersteuern des Optimizer Mode auf Session Ebene:
ALTER SESSION SET OPTIMIZER_GOAL = RULE;
ALTER SESSION SET OPTIMIZER_GOAL = CHOOSE;
Übersteuern des Optimizer Mode auf Befehls Ebene:
SELECT /*+ RULE */ ename FROM .....
SELECT /*+ CHOOSE */ ename FROM ....
Ausgabe 5. Januar 1995
Seite 5Seite
von 82
-5
$NDGLD $*
Oracle Datenbank Tuning
COST based
Optimizer
Der COST based Optimizer legt den Execution Plan aufgrund von Statistiken fest. Diese Statistiken müssen regelmässig erstellt werden
(Vorteilhaft während der Nacht mittels ANALYZE ... Kommando).
Vorteile des COST Based Optimizers
• Der CBO erkennt unnötige Abfragen, wie sie oft von Endanwender
Tools generiert werden.
• Umschaltung von «LIKE» auf «=» wenn Wildcard Zeichen fehlen.
• Verwendung von neuen Features wie Hash-Join, Anti-Join etc.
• Entscheidet sich nicht «stur» für einen Indexzugriff wie dies der RULE
based Optimizer oft tut.
Nachteile des COST Based Optimizers
• Schätzt die Selektivität der Daten manchmal falsch ein, was dann fast
immer zu einem Fulltable-Scan führt.
• Der CBO darf erst wirklich produktiv ab Version 7.3 eingesetzt werden.
• Der Parse Aufwand ist grösser als beim RULE based Optimizer.
Entscheidungsgrundlagen
Der COST based Optimizer benutzt folgende Entscheidungsgrundlagen:
ANALYZE ....
USER_TABLES
Histogramm
USER_TAB_COLUMNS
DB_FILE_MULTIBLOCK_READ_COUNT
Dictionary
COST Based
Optimizer
Execution Plan
• Statistiken werden im Data Dictionary abgelegt. Sie können eingesehen werden in den Views USER_TABLES und USER_TAB_COLUMNS.
• Berücksichtigung der Selektivität, wenn ein Histogramm zur Verfügung
steht. Die Selektivität ist der Anteil an Rows in Prozent, die das Query
selektiert. Queries, welche wenig Rows selektieren, haben eine gute
Selektivität, sie eignen sich für einen Index-Scan.
• Der INIT.ORA Parameter DB_FILE_MULTIBLOCK_READ_COUNT
wird berücksichtigt. Hohe Werte verleiten den COST Based Optimizer
zu einem Fulltable-Scan.
Seite
6 von
Ausgabe
5.82
Januar
1995
Seite -6
$NDGLD $*
Oracle Datenbank Tuning
2.2
Zugriffsmethoden
Full Table Scan
Wenn sich der Optimizer für einen Full Table Scan entscheidet, so wird
bei dieser Zugriffsmethode die gesamte Tabelle sequentiell gelesen.
Full Table Scans sind vorteilhaft, wenn eine grosse Datenmenge als Endresultat gesucht wird.
Tabelle
sequentiell
lesen
Als Richtwerte können folgende Aussagen gemacht werden:
• Datenmengen > 15 % werden in der Regel effizienter mit einem Full
Table Scan gelesen. Die genaue Grenze ist jedoch abhängig vom Clustering Factor (Datenverteilung), der Datendichte pro Block, den Anzahl Attributen etc.
• Oracle hat einen sehr effizienten Mechanismus für Full Table Scans. Es
wird ein sogenannter Multi-Block-Read gelesen, welcher pro Disk Operation mehrere Blocks lesen kann.
• Der Durchsatz des Multi-Block-Reads wird mit dem INIT.ORA Parameter DB_FILE_MULTIBLOCK_READ_COUNT definiert.
• Einerseits sollten möglichst viele Blocks gelesen werden (I/O wird kleiner), andererseits sollten auch nicht zuviele Blocks angefordert werdsen, so dass während der Wartezeit bis die Blocks gefunden werden,
die CPU arbeitslos ist.
• Einen zu hohen DB_FILE_MULTIBLOCK_READ_COUNT verleitet den
COST based Optimizer zu Full Table Scans. Deshalb muss dieser Wert
sorgfältig evaluiert werden.
Kontrolle von
DB_FILE_MULTIBLOCK
READ_COUNT
SELECT NAME "File",PHYBLKWRT WRITTEN,PHYBLKRD READ,
ROUND(PHYBLKRD/DECODE(PHYRDS,0,1,PHYRDS),2) "Ratio"
FROM V$DBFILE A, V$FILESTAT B
WHERE A.FILE# = B.FILE#
ORDER BY SUBSTR(NAME,1,INSTR(NAME,’:’));
File
WRITTEN
READ
Ratio
-------------------- ---------- ---------- ---------/sipp/d1/ts_system/P
87378
86262
1,91
/dev/vg11/rsipp01
1500215
2651
1
/dev/vg11/rsipp02
2593390
1892911
4,65
/dev/vg11/rsipp08
6638830
6007879
1,57
/sipp/d1/ts_system/P
0
0
0
/dev/vg11/rsipp03
3199930
16049439
6,42
/dev/vg11/rsipp04
189468
23607191
15,02
/dev/vg11/rsipp05
60560
8838645
8,37
/dev/vg11/rsipp06
504939
637395
1,03
/dev/vg11/rsipp07
3132762
1838936
1,25
/dev/vg11/rsipp09
27764
1210526
13,77
/dev/vg11/rsipp10
9641
420278
13,9
Ausgabe 5. Januar 1995
Seite 7Seite
von 82
-7
$NDGLD $*
Oracle Datenbank Tuning
Zuordnung DB-File und Tablespace
Man erkennt, dass der maximale Durchsatz nur auf dem grossen Tablespace TAB_CDR (/dev/vg11/rsipp04, /dev/vg11/rsipp09, /dev/vg11/rsipp10)
erreicht wird. Ferner sieht man, dass ein Index Scan diesen Mechanismus nicht verwendet IDX_SIPP und IDX_BIG (/dev/vg11/rsipp06 und
/dev/vg11/rsipp07).
Index Scan
Bei einem Index Scan werden die Daten aufgrund der WHERE Klausel
zuerst im Index lokalisiert, dort wird die ROWID des Treffers bestimmt
und dann der entsprechende Datenblock direkt gelesen. Die Adresse des
Datenblocks ist in der ROWID enthalten.
Jede Row besitzt eine eindeutige physikalische Adresse, welche durch
ROWID repräsentiert wird. ROWID ist aus drei Teilen aufgebaut:
Aufbau ROWID
Nr des Blocks
im File
Nr der Row
im Block
Nr des Files
SELECT rowid, ename FROM emp;
ROWID
-----------------000005DF.0000.0004
000005DF.0001.0004
000005DF.0002.0004
000005DF.0003.0004
000005DF.0004.0004
000005DF.0005.0004
000005DF.0006.0004
000005DF.0007.0004
000005DF.0008.0004
000005DF.0009.0004
000005DF.000A.0004
ENAME
---------SMITH
ALLEN
WARD
JONES
MARTIN
BLAKE
CLARK
SCOTT
KING
TURNER
ADAMS
Alle Rows sind im gleichen File
Nummer der Row im Block
Alle Rows sind im gleichen Block, kein «Block Chaining»
Oracle kennt zwei Arten von Indexen, den B*TREE Index und den Bitmapped Index.
Seite
8 von
Ausgabe
5.82
Januar
1995
Seite -8
$NDGLD $*
Oracle Datenbank Tuning
B*TREE Index
B*TREE Indexe sind nach dem Datenwert sortierte baumförmige Indizes.
Man spricht auch von einem sorted Index.
B-Trees bestehen aus Index- und Sequenzblöcken. In den Sequenzblökken werden die ROWID’s der Records gespeichert:
Index-Blöcke
Sequenz-Blöcke als «Doubly-Linked List»
Index-Blöcke:
Baumstrukturierter Index-Satz zu den Sequenzblöcken
Sequenz-Blöcke: «Leafs», enthalten die ROWID’s
Unique und Nonunique Indexes
Wird eine Column mit einem Unique Index versehen, dann muß jeder
Wert der Column eindeutig sein. Dies trifft bei Primary Keys zu, auf diese
wird immer ein Unique Index angelegt, was unter Oracle7 automatisch
erfolgt. Doch auch nicht eindeutige Columns können indexiert werden,
indem beim Kreieren des Index das Schlüsselwort «unique» angegeben
wird. Das Anlegen von Indexes auf non-unique Columns ist jedoch nur
dann sinnvoll, wenn der Wertebereich groß ist. Eine Column, die z.B. nur
die Werte «männlich», «weiblich» beinhalten kann, ist denkbar ungeeignet zum indexieren !
Concatenated Index
Aus mehreren Columns bestehende Indexe werden «Concatenated
Index» genannt. Sie sind sehr ideal, wenn Eindeutigkeit über mehrere
Columns gefordert ist.
Vorwahl
Telefon
PLZ
Ort
Straße
033
345 02 40
3628
Seftigen
Sonnenrain 5
031
385 30 11
3000
Bern
Zieglerstr. 34
Die Columns «Tel.-Vorwahl» und «Telefon» sind zusammen unique und
deshalb für einen concatenated Index geeignet.
Ausgabe 5. Januar 1995
Seite 9Seite
von 82
-9
$NDGLD $*
Oracle Datenbank Tuning
Verwendung
des Index
B*TREE Indexes erlauben eine effiziente Suche, wenn die indexierten
Daten selektiv sind. Die selektivste Indexierung ist die Sonderform eines
Unique Index, weil es dort genau 1 einzige Row pro Datenwert gibt.
Ein B*TREE Index kann in den folgenden Fällen nicht verwendet werden:
• NULL Werte sind nicht im Index gespeichert und können nur in der
Tabelle mittels Full Table Scan gefunden werden.
... WHERE column IS NULL;
... WHERE column IS NOT NULL;
(Index)
(Index)
• Alle Funktionen direkt auf der Column verhindern den Indexzugriff
... WHERE SUBSTR(ename,1,7) = ’CAPITAL’;
... WHERE ename LIKE ’CAPITAL’;
(Index)
(Index OK)
... WHERE TRUNC(edate) = TRUNC(sysdate)
... WHERE edate BETWEEN TRUNC(sysdate)
AND TRUNC(sysdate) + .99999;
(Index)
(Index OK)
• Ungleich Operator verhindert Indexzugriff
... WHERE amount != Wildcard;
... WHERE amount > 0;
(Index)
(Index OK)
• Berechnungen verhindern den Indexzugriff
... WHERE amount + 3000 < 5000;
... WHERE amount < 2000;
(Index)
(Index OK)
• Wildcard % auf erstem Zeichen verhindert Indexzugriff
... WHERE ename LIKE ’%ueller’;
(Index)
• Gleiche Columns auf beiden Seiten verhindern Indexzugriff
... WHERE ename = NVL(var, ename);
... WHERE ename LIKE NVL(var, ’%’);
(Index)
(Index OK)
• String Concatenierung verhindert den Indexzugriff
... WHERE ename || vname = ’XYZ’;
... WHERE ename = ’XY’ AND vname = ’Z’;
(Index)
(Index OK)
• Mehr als ein Index (auf ename und deptno) pro Tabelle kann nur verwendet werden, wenn alles Equality-Searches mit NON-UNIQUE
Indexes sind. Hier führt Oracle einen Index-Merge durch.
... WHERE ename LIKE ’XY%’ AND deptno = 10;
... WHERE ename = ’XY’ AND deptno = 10;
Seite
10 von
82
Ausgabe
5. Januar
1995
(ein Index)
(beide Indexe)
Seite -10
$NDGLD $*
Oracle Datenbank Tuning
• Oracle erlaubt einen Index über mehrere Attribute in der gleichen Tabelle zu erstellen. Ein solcher concatenated Index eröffnet dem Optimizer ideale Zugriffe, die von zwei einzelnen Indexes nicht zu lösen sind.
Im folgenden Beispiel ist ein Index über (deptno, ename) vorhanden.
SELECT
FROM
WHERE
AND
ename, sal, deptno
emp
ename LIKE ’KIN%’
deptno = 10;
(Index OK)
SELECT ename, sal, deptno
FROM emp
WHERE deptno = 10;
(Index OK)
SELECT ename, sal, deptno
FROM emp
WHERE ename LIKE ’KIN%’;
(Index)
Im dritten Beispiel kann der Index nicht benutzt werden, da das Attribut
ename im Index an zweiter Stelle steht.
• NOT verhindert den Indexzugriff
SELECT ename, sal, deptno
FROM emp
WHERE deptno NOT = 0
(Index)
SELECT ename, sal, deptno
FROM emp
WHERE deptno > 0
(Index OK)
• Manchmal will man einen Indexzugriff explizit verhindern. Dies wird mit
einer Addition von 0 oder einer Concatenierung eines Leerstrings erreicht.
Ausgabe 5. Januar 1995
SELECT ename, sal, deptno
FROM emp
WHERE deptno + 0 = 10
(Index)
SELECT ename, sal, deptno
FROM emp
WHERE ename || ’’ = ’MU%’
(Index)
Seite 11
von-11
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
Bitmapped Index
Ab Oracle Version 7.3 stehen auch Bitmapped Indexe zur Verfügung. Sie
eignen sich hervorragend, um unselektive Attribute zu indexieren, das
heisst es gibt nur wenige DISTINCT Werte.
Eigenschaften von Bitmapped Indexen
• Nur der neue Optimizer CBO kann Bitmapped Indexe verwenden.
• Verknüpfte Bitmapped Indexes sind möglich.
• NULL Werte können ebenfalls im Bitmapped Index gefunden werden.
Das heisst also, das Abfragen auf NULL und IS NULL möglich sind.
• Bitmapped Indexe können auch != Abfragen lösen.
• Bitmapped Indexe eignen sich für SELECT COUNT(*) Abfragen.
• Das Nachführen von Bitmapped Indexen ist aufwendig.
• Das Locking eines Bit ist nicht möglich, deshalb lockt Oracle einen gesamten Index-Block, was zu Locking Problemen führen kann. Bitmapped Indexe sollten also in OLTP Datenbanken mit gleichzeitigen DML
Befehlen auf der gleichen Tabelle nicht eingesetzt werden. In DataWareHouse Anwendungen bieten sie sehr grosse Vorteile.
Beispiel: Abfragen auf männlich, weiblich sind sehr unselektiv
ROWID
A
B
C
D
E
M
W
1
0
1
0
0
1
1
0
1
0
Einsatz von Bitmapped Indexen
Bitmapped Indexe eignen sich gut für nicht selektive Attribute. Man muss
also zuerst die Selektivität feststellen.
SELECT status, count(*) FROM credit
GROUP BY status;
STATUS COUNT(*)
--------- --------0
176858
1
4
2
373
In diesem Fall haben wir lediglich 3 verschiedene Werte, ein Bitmapped
Index könnte in Betracht gezogen werden.
SELECT lname, count(*) FROM customer
GROUP BY lname;
LNAME
COUNT(*)
---------------------------------------- --------AST Communication
3
Abbet
1
Abbondioli
1
.....
In diesem Fall kommt ein Bitmapped Index nicht in Frage.
Seite
12 von
82
Ausgabe
5. Januar
1995
Seite -12
$NDGLD $*
Oracle Datenbank Tuning
2.3
Indexierungs Strategie
Regeln
• Grundsätzlich werden alle Primary Key und UNIQUE Keys mittels
Constraint indexiert.
• Fremdschlüssel werden meistens indexiert wenn eine Referential
Integrity implementiert ist. Fremdschlüssel von Definitionstabellen
(Lookup-Tables) müssen nicht indexiert werden, da auf diesen Tabellen keine DDL-Kommandos (UPDATE, INSERT, DELETE) durchgeführt werden und somit kein Locking eintreten kann.
• Intersection Tabellen werden meistens mit 2 Indexen in beiden Richtungen indexiert.
MSISDN
(ms_id)
SIM
(imsi)
BINDING
(ms_id, imsi)
Primary Key (ms_id, imsi)
UNIQUE Index (imsi, ms_id)
• Alle Suchfelder (WHERE Klausel), welche eine gute Selektivität aufweisen, werden indexiert. Suchfelder müssen bekannt sein (Oracle Expert,
Entwickler Interviews etc).
• Suchfelder, welche eine gute Selektivität aufweisen, jedoch unregelmässig verteilte Datenwerte haben, werden zusätzlich mit einem Histogramm versehen.
• Auf jeder Tabelle muss kontrolliert werden, dass nicht zwei Indexe mit
dem gleichen Attribut beginnen.
• Bestehende Indexe müssen analysiert werden (ANALYZE INDEX).
Selektivität
Die Selektivität kann wie folgt festgestellt werden (Beispiel):
SELECT SUBSTR(TO_CHAR(date_cdr1,’DDMMYYYY’),1,8)
"Wert", count(*) "Anzahl"
FROM acm
GROUP BY TO_CHAR(date_cdr1,’DDMMYYYY’)
ORDER BY 2 desc;
Wert
Anzahl
-----------------23051997
649
22051997
604
24051997
578
02051997
573
21051997
536
17051997
535
09051997
533
03051997
527
16051997
515
01051997
502
30041997
496
Ausgabe 5. Januar 1995
Pro Tag (date_cdr1) werden zwischen 400
und 650 Reloads durchgeführt. Die Selektivität ist nicht hervorragend aber für einen
Index-Scan auf einen bestimmten Tag durchaus noch akzeptabel.
Seite 13
von-13
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
B*TREE Index
Analyse
Um ganz sicher zu sein, ob ein Index sinnvoll ist oder nicht, kann eine
Index Analyse durchgeführt werden. Dazu wird der Index analysiert und
anschliessend wird das erstellte Index-Histogramm kontrolliert.
ANALYZE INDEX cdr_imei_btidx VALIDATE STRUCTURE;
SELECT repeat_count "R-Count",
keys_with_repeat_count "Keys"
FROM index_histogram;
R-Count
Keys
---------- ---------1
63163
512
593
1024
115
1536
26
2048
9
2560
5
3072
1
3584
1
4096
1
4608
0
5120
0
5632
0
6144
0
6656
0
7168
0
7680
0
Auswertung
Es gibt 63’163 IMEI-Werte, welche genau ein einziges Mal vorkommen in
der Tabelle CDR. Für diese 63’163 IMEI’s ist ein Indexzugriff sehr
effizient, da die Selektivität sehr gut ist. Wir haben jedoch auch einen
IMEI-Wert, der 3072 mal vorkommt, wird also nach diesem IMEI-Wert
selektiert, dann ist ein Indexzugriff nicht vorteilhaft. Welche IMEI-Werte
kommt denn so oft vor ?
SELECT
FROM
GROUP
HAVING
imei
CDR
BY imei
COUNT(*) > 3070;
IMEI
--------------490137201144390
490137203883610
490137203905860
Seite
14 von
82
Ausgabe
5. Januar
1995
Seite -14
$NDGLD $*
Oracle Datenbank Tuning
2.4
Datenverteilung
Der COST Based Optimizer geht normalerweise von einer gleichmässigen Datenverteilung aus, er kennt also nur die Anzahl Rows und die
Anzahl unterschiedlicher Datenwerte (distinct Keys). Diese Werte werden
ihm mittels ANALYZE jede Nacht zur Verfügung gestellt.
Bei der folgenden Abfrage ist damit zu rechnen, dass nicht die gleiche
Selektivität vorhanden ist:
SELECT ename, sal FROM emp
WHERE ename = ’MEIER’;
SELECT ename, sal FROM emp
WHERE ename = ’VAN NIEVERGELT’;
Typischerweise findet man oft folgenden Sachverhalt:
• Die häufigsten Datenwerte werden auch am häufigsten abgefragt
• Die häufigsten Datenwerte werden am wenigsten abgefragt.
In beiden Fällen besteht das Dilemma, dass für einige Abfragen ein Index
vorteilhaft, für andere Abfragen ein Full-Table Scan besser wäre. Genau
dieses Problem lösen ab Oracle 7.3 Histogramme.
Histogramme zeigen die Verteilung und die Häufigkeit der Datenwerte
auf. Damit kann der CBO entscheiden, ob er einen Index-Scan oder
einen Fulltable-Scan anwenden soll.
Selektivität
Viele unterschiedliche Werte, gute Selektivität
Index-Scan ideal
Wenig unterschiedliche Werte, schlechte Selektivität
Fulltable-Scan ideal
Bereich
Histogramme werden mittels ANALYZE ... FOR ... erstellt. Sie können mit
der View USER_HISTOGRAMS dargestellt werden.
Bucket
Endwert
2
3
4
5
6
7
8
10 20
30
40
50
60
70
80
1
9
10
90 100
Gleichmässige Datenverteilung: 10 % der Daten sind in Bucket 10
Bucket
Endwert
1
5
2
5
3
5
4
5
5
6
7
8
10
20
50
60
9
10
70 100
Ungleichmässige Datenverteilung: 30 % der Daten müssen sich Bucket 10 teilen
Ausgabe 5. Januar 1995
Seite 15
von-15
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
2.5
Join Algorithmen
Oracle kennt verschiedene Join-Methoden, um zwei Mengen miteinander
zu verbinden. Jede Methode hat Vorteile, welche typischerweise von der
Datenmenge abhängig ist.
Sort-Merge Join
Beim Sort-Merge Join werden die Tabellen über das Sort-Modul aufgrund
der Join Bedingung sortiert. Allfällige WHERE Klauseln können direkt
während dem Full Table Scan evaluiert werden und schicken dementsprechend weniger Treffer in die nächst höhere Row Source.
Grundsätzlich sollte die kleinere Tabelle zuerst gelesen werden. Die kleinere Tabelle ist diejenige, welche nach Berücksichtigung aller WHEREKlauseln weniger Rows in die nächst höhere Row Source schickt. SortMerge Joins eignen sich für grosse Datenmengen, sie können nur bei
Equi-Joins verwendet werden.
Nachteil des Sort-Merge: Sortiervorgang, obwohl dies durch Oracle sehr
effizient gemacht wird.
Beispiel
SELECT e.ename, e.sal, d.dname
FROM emp e, dept d
WHERE e.deptno = d.deptno;
Execution Plan
----------------------------------------------------SELECT STATEMENT Optimizer=CHOOSE
MERGE JOIN
SORT (JOIN)
TABLE ACCESS (FULL) OF ’EMP’
SORT (JOIN)
TABLE ACCESS (BY ROWID) OF ’DEPT’
Darstellung des Execution Plans
Sort-Merge Algorithmus
Merge-Join
Sort
Emp (Full)
Sort
Sort-Modul
Dept (Full)
Driving Table
(Hier wird begonnen)
Seite
16 von
82
Ausgabe
5. Januar
1995
Seite -16
$NDGLD $*
Oracle Datenbank Tuning
Hash-Join
Hash-Joins sind eine neue Möglichkeit, welche ab Oracle 7.3 angeboten
werden. Sie stehen in direkter Konkurrenz zu Sort-Merge Joins und sind
diesen meistens auch überlegen. Hash-Joins eignen sich ebenfalls für
grosse Datenmengen. Die Zielsetzung des Hash-Joins ist es, den Sort
durch einen effizienteren Mechanismus zu ersetzen. Hashing ist die Festlegung eines Wertes durch eine Berechnung. Hash-Joins werden nur
vom CBO unterstützt, der RBO kennt diesen Algorithmus nicht. HashJoins funktionieren nur auf Equi-Joins. Oracle bildet mit der kleineren
Tabelle eine Hash-Table und gleicht danach die grössere Tabelle gegen
diese Hash-Table ab.
Beispiel
SELECT e.ename, e.sal, d.dname
FROM emp e, dept d
WHERE e.deptno = d.deptno;
Execution Plan
----------------------------------------------------SELECT STATEMENT Optimizer=ALL_ROWS
HASH JOIN
TABLE ACCESS (FULL) OF ’EMP’
TABLE ACCESS (FULL) OF ’DEPT’
Darstellung des Execution Plans
Hash-Join
Emp (Full)
Dept (Full)
Generiert Hash-Table
im Memory
Prüfung der Werte
gegen Hash-Table
Driving Table
(Hier wird begonnen)
Der Hash-Join wird ab Version 7.3 dem Sort-Merge Join bevorzugt eingesetzt. Um den Hash-Join zu aktivieren, müssen die entsprechenden
INIT.ORA Parameter gesetzt sein, dies sind insbesondere die Parameter
HASH_JOIN_ENABLED und HASH_AREA_SIZE.
Ausgabe 5. Januar 1995
Seite 17
von-17
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
Nested Loop
Bei diesem Algorithmus werden zwei verschachtelte Loops verwendet.
Für jeden Zugriff im äusseren Loop wird einmal der innere Loop ausgeführt. Die Kosten wachsen somit sehr schnell, womit ein Nested Loop nur
für kleine Datenmengen effizient ist. Die zu verarbeitende Datenmenge
wird von äusseren Loop bestimmt. Die kleinere Tabelle sollte deswegen
im äusseren Loop stehen. Dies ist wiederum die Tabelle, welche nach
Berücksichtigung aller WHERE Klauseln weniger Rows in die nächst
höhere Row Source schickt.
• Für kleinere Datenmengen.
• Ideal wenn WHERE Klausel Einschränkung enthält.
• Zu verarbeitende Datenmenge ist vom äusseren Loop abhängig.
Algorithmus
Algorithmus des Nested Loop
BEGIN
FOR outer_loop IN (SELECT deptno FROM dept)
LOOP
FOR inner_loop IN (SELECT empno FROM emp
WHERE deptno = outer_loop.deptno)
LOOP
RETURN;
END LOOP inner_loop;
END LOOP outer_loop;
END;
Beispiel
SELECT
FROM
WHERE
AND
e.ename, e.sal, d.dname
emp e, dept d
d.deptno = e.deptno
d.dname = ’SALES’;
Execution Plan
---------------------------------------------------SELECT STATEMENT Optimizer=CHOOSE
NESTED LOOPS
TABLE ACCESS (FULL) OF 'DEPT'
TABLE ACCESS (BY ROWID) OF 'EMP'
INDEX (RANGE SCAN) OF 'EMP_DETNO'
Darstellung des Execution Plans
(Inner Loop)
SELECT ... FROM emp
WHERE deptno = 20;
Nested Loop
Index Scan
Emp(deptno)
Emp by
ROWID
Dept (Full)
Driving Table mit Restriktion: WHERE dname = ’SALES’
Kann Index oder Full Table Scan sein.
(Outer Loop)
Seite
18 von
82
Ausgabe
5. Januar
1995
Seite -18
Oracle Datenbank Tuning
$NDGLD $*
Star Joins
Normalerweise versucht der Optimizer um jeden Preis ein Kartesisches
Produkt zu vermeiden und joint nur Tabellen, welche aufgrund der
WHERE Klausel auch wirklich joinbar sind. Bei sehr grossen Tabellen,
welche vor allem in Decision Support Systemen (DSS) oder auch DataWareHouses vorkommen, bietet Oracle 7.3 eine spezielle Join-Methode
an, den Star Join.
Bei einem Star-Join wird eine sehr grosse zentrale Tabelle (Fact Table)
geschaffen, welche mit mindestens 2 weiteren kleinen Tabellen einen
Join bilden.
Beurteilung
• Sort-Merge Join und Hash-Join eignen sich für grosse Datenmengen,
wo der Gesamtdurchsatz eine wichtige Rolle spielt.
• Der neuere Hash-Join ist dem älteren Sort-Merge Join fast immer überlegen und sollte ab Oracle 7.3 bevorzugt eingesetzt werden.
• Nested Loops sind sehr effizient bei kleineren Datenmengen sowie
einer WHERE Klausel im äusseren Loop.
Ausgabe 5. Januar 1995
Seite 19
von-19
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
2.6
Execution Plan
Da der COST Based Optimizer seine Entscheidungen aufgrund vieler
komplexer Abhängigkeiten fällt, ist es praktisch unmöglich, eine Vorhersage seiner Zugriffe zu machen. Oracle stellt deshalb einige Tools zur
Verfügung, welche es erlauben, den Execution Plan zu visualisieren und
zu beeinflussen (Hints).
Folgende Tools stehen zur Verfügung:
AutoTrace in
SQL*Plus
Damit kann sehr rasch und einfach der Execution Plan kontrolliert werden. Ferner werden Angaben gemacht über die «Kostenberechnung»
des Optimizer. Man kann mit diesem Hilfsmittel sehr rasch feststellen, ob
sich der Optimizer «verschätzt» hat. Man sieht jedoch die tatsächlich verarbeiteten Datenmengen in den einzelnen Modulen nicht. Für genauere
Abklärungen muss der umständlichere Weg über TKPROF gewählt
werden.
Installation von
AutoTrace
Nur notwendig, wenn dies nicht bereits gemacht wurde.
1. Script utlxplan.sql als User SYS laufen lassen. Dieses Script erstellt
die notwendige Tabelle PLAN_TABLE. Dieses Script ist im HOME Directory von Oracle zu finden.
2. Die PLAN_TABLE muss an alle User die AutoTrace nutzen dürfen verfügbar gemacht werden.
GRANT ALL ON SYS.PLAN_TABLE TO ...;
CREATE PUBLIC SYNONYM PLAN_TABLE FOR SYS.PLAN_TABLE;
3. Rolle «PLUSTRACE» erstellen mit dem Script plustrce.sql. Dieses
Script ist im HOME Directory von SQL*PLUS zu finden.
4. Die Rolle «PLUSTRACE» an alle User die AutoTrace nutzen dürfen
granten.
GRANT PLUSTRACE TO ...;
Nutzen von
AutoTrace
Innerhalb von SQL*PLUS können folgende Optionen gesetzt werden:
SET AUTOTRACE {OFF | ON | TRACEONLY} [EXPLAIN] [STATISTICS]
• ON, OFF, TRACEONLY: Ein-, Ausschalten der ganzen Option oder
nur Execution Plan anzeigen.
• EXPLAIN
Execution Plan anzeigen.
• STATISTICS
Statistiken wie Netzwerkverkehr anzeigen.
Beispiele
SQL> set autotrace on explain
SQL> select * from dept;
DEPTNO
--------10
20
DNAME
-------------ACCOUNTING
RESEARCH
LOC
------------NEW YORK
DALLAS
Execution Plan
---------------------------------------------------------SELECT STATEMENT Optimizer=CHOOSE (Cost=1 Card=4 Bytes=140)
TABLE ACCESS (FULL) OF ’DEPT’ (Cost=1 Card=4 Bytes=140)
Seite
20 von
82
Ausgabe
5. Januar
1995
Seite -20
$NDGLD $*
Oracle Datenbank Tuning
SQL> set autotrace on explain statistics
SQL> select * from dept;
DEPTNO
--------10
20
DNAME
-------------ACCOUNTING
RESEARCH
LOC
------------NEW YORK
DALLAS
Execution Plan
---------------------------------------------------------SELECT STATEMENT Optimizer=CHOOSE (Cost=1 Card=4 Bytes=140)
TABLE ACCESS (FULL) OF ’DEPT’ (Cost=1 Card=4 Bytes=140)
Statistics
----------------------------------------------------0 recursive calls
2 db block gets
1 consistent gets
0 physical reads
0 redo size
386 bytes sent via SQL*Net to client
288 bytes received via SQL*Net from client
3 SQL*Net roundtrips to/from client
0 sorts (memory)
0 sorts (disk)
4 rows processed
SQL> set autotrace traceonly explain
SQL> select * from dept;
Execution Plan
---------------------------------------------------------SELECT STATEMENT Optimizer=CHOOSE (Cost=1 Card=4 Bytes=140)
TABLE ACCESS (FULL) OF ’DEPT’ (Cost=1 Card=4 Bytes=140)
AutoTrace ist erst ab Oracle 7.3 verfügbar. Es eignet sich hervorragend
für rasche ad hoc Kontrollen. Es empfiehlt sich deshalb sehr, die eigenen
SQL-Statements auf diese Weise rasch zu kontrollieren und allenfalls mittels HINT’s zu tunen. Um auch eine Zeitmessung in SQL*PLUS
anzuzeigen, kann die Option SET TIMING ON zusätzlich eingeschaltet
werden.
Angaben im
Execution Plan
Optimizer: Angabe welcher Optimizer gewählt wurde.
Für den CBO werden zusätzlich noch die folgenden Angaben gemacht:
Cost:
Card:
Bytes:
Ausgabe 5. Januar 1995
«Kostenberechnung» des CBO
Cardinality bezeichnet die geschätzte Anzahl Rows, welche
auf jeder Stufe (Row-Source) zu verarbeiten sind. Diese Information ist sehr hilfreich, da sich der CBO ja aufgrund von
Statistiken entscheidet. Fehlentscheide sind somit sehr
schnell ersichtlich. Die Cardinality ist also eine Schätzung des
CBO, wieviel Arbeit (Anzahl Rows) auf jeder Stufe zu verrichten sind.
Schätzung der zu verarbeitenden Bytes.
Seite 21
von-21
82
Seite
Oracle Datenbank Tuning
TopSession im
Enterprise Manager
$NDGLD $*
Das aktuelle SQL-Statement einer beliebigen Session kann sehr rasch
und auf einfache Weise analysiert werden.
Session
auswählen
Details der
Session
Execution Plan
der Session
Seite
22 von
82
Ausgabe
5. Januar
1995
Seite -22
$NDGLD $*
Oracle Datenbank Tuning
TKPROF
Genaue Analyse des SQL-Statements anhand eines generierten Tracefiles. Dazu muss das SQL-Tracing eingeschaltet werden und
anschliessend wird das erzeugte Tracefile analysiert.
SQL-Tracing
einschalten
TKPROF
SQL ausführen
SQL-Tracing
ausschalten
Installation von
TKPROF
Trace-File
Output
Nur notwendig, wenn dies nicht bereits gemacht wurde.
1. Stored Procedure trace als User SYS installieren. Damit wird der Name
des erzeugten Trace-Files angezeigt.
CREATE OR REPLACE
PROCEDURE trace (pOnOff IN BOOLEAN) IS
vDirectory
vFile
vFileName
vMode
vGlobal_name
vSPid
vServer
vDate
v$parameter.value%TYPE;
utl_file.file_type;
VARCHAR2(9):= ’trace.log’;
VARCHAR2(1):= ’a’;-- append
global_name.global_name%TYPE;
v$process.spid%TYPE;
v$session.server%TYPE;
VARCHAR2(21) := TO_CHAR(SYSDATE,’DD.MM.YYYY HH24:MI.SS’);
/*
* Internal function to find the first valid directory for utl_file.
* There can be more than one.
* If there is none, then the fopen will fail..., so make sure that
* there is one...
*/
FUNCTION utl_file_dir
RETURN VARCHAR2 IS
BEGIN
BEGIN
SELECT value
INTO vDirectory
FROM v$parameter
WHERE name = ’utl_file_dir’;
EXCEPTION
WHEN NO_DATA_FOUND THEN
vDirectory := NULL;
END;
IF (INSTR(vDirectory,’,’) > 0) THEN
vDirectory := SUBSTR(vDirectory,1,INSTR(vDirectory,’,’)-1);
END IF;
RETURN(LTRIM(RTRIM(vDirectory)));
END utl_file_dir;
Ausgabe 5. Januar 1995
Seite 23
von-23
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
BEGIN
IF (pOnOff) THEN
BEGIN
SELECT g.global_name,p.spid,s.server
INTO vGlobal_name, vSPid, vServer
FROM v$session s, v$process p, global_name g
WHERE s.paddr = p.addr
AND s.audsid = USERENV(’sessionid’);
END;
BEGIN
vFile := utl_File.FOpen(utl_file_dir,vFilename,vMode);
EXCEPTION
WHEN utl_File.invalid_operation THEN
/* The file does not yet exist, so create it...*/
vFile := utl_File.FOpen(utl_file_dir,vFilename,’w’);
END;
utl_File.putf(vFile,’At %s user %s@%s opens file %s with %s server \n’,
vDate, USER,vGlobal_name,vSPid,vServer);
utl_File.fclose(vFile);
dbms_output.put_line(’You have server ’||vSPid||’ (’||vServer||’)’);
END IF;
DBMS_Session.set_sql_trace(pOnOFF);
END;
/
2. Identifikation des Trace Directory, definiert im Parameter
USER_DUMP_DEST im File INIT.ORA. Fragen Sie am besten den
DB-Verantwortlichen nach dem Ort und den Berechtigungen dieses
Directory. Für das PBB Projekt lautet dieses Directory:
GD3I9K: /sipp/oracle/admin/P18M7S/udmp
GD3IQB: /sipp/oracle/admin/T18M7S/udmp
Nutzen von
TKPROF
Innerhalb von SQL*PLUS wird Tracing ein- und ausgeschaltet:
SQL> set serveroutput on
SQL> execute trace(true);
You have server 24539 (DEDICATED)
SQL> SELECT e.ename, e.sal, d.dname
FROM emp e, dept d
WHERE d.deptno = e.deptno
AND d.dname = ’SALES’;
SQL> execute trace(false);
Nun in das Trace Directory wechseln
$ cd /sipp/oracle/admin/T18M7S/udmp
Nun wird das Outputfile trace.lis generiert
$ tkprof ora_24539.trc trace.lis explain=user/pwd sort=exeela sys=no
explain=user/pwd
sys=no
exeela
Seite
24 von
82
Ausgabe
5. Januar
1995
Connect to ORACLE and issue EXPLAIN PLAIN.
TKPROF does not list SQL statements run as user SYS.
elapsed time executing
Seite -24
$NDGLD $*
Oracle Datenbank Tuning
Output-File
Das Outputfile von TKPORF hat den folgenden Inhalt
TKPROF: Release 7.3.2.2.0 - Production on Sat Jun 7 15:12:15 1997
Copyright (c) Oracle Corporation 1979, 1994. All rights reserved.
Trace file: ora_26996.trc
Sort options: exeela
*******************************************************************************
count
= number of times OCI procedure was executed
cpu
= cpu time in seconds executing
elapsed = elapsed time in seconds executing
disk
= number of physical reads of buffers from disk
query
= number of buffers gotten for consistent read
current = number of buffers gotten in current mode (usually for update)
rows
= number of rows processed by the fetch or execute call
*******************************************************************************
SELECT e.ename, e.sal, d.dname
FROM emp e, dept d
WHERE d.deptno = e.deptno
AND d.dname = ’SALES’
call
count
cpu
elapsed
disk
query
current
rows
------- ------ -------- ---------- ---------- ---------- ---------- ---------Parse
1
0.01
0.01
0
0
0
0
Execute
1
0.00
0.00
0
0
0
0
Fetch
1
0.01
0.02
2
3
4
6
------- ------ -------- ---------- ---------- ---------- ---------- ---------total
3
0.02
0.03
2
3
4
6
Misses in library cache during parse: 1
Optimizer goal: CHOOSE
Parsing user id: 28 (SCOTT)
Rows
Execution Plan
------- --------------------------------------------------0 SELECT STATEMENT GOAL: CHOOSE
6 NESTED LOOPS
4
TABLE ACCESS GOAL: ANALYZED (FULL) OF ’DEPT’
14
TABLE ACCESS GOAL: ANALYZED (FULL) OF ’EMP’
Auswertung
Ausgabe 5. Januar 1995
• Von grosser Wichtigkeit sind die Anzahl Rows, welche auf jeder Stufe
verarbeitet werden. Hier ist darauf zu achten, dass eine Restriktion der
Datenmenge aufgrund der WHERE Klausel so früh als möglich eintritt.
• Sehr hohe Zahlen bei «Query» deuten auf eine Repetition der gleichen
Arbeit hin. Insbesondere bei ungünstigen Nested Loops ist «Query»
oftmals sehr gross und Einsparungen sind möglich.
• «CPU» und «ELAPSED» sollten möglichst identisch sein, dann liegt
kein Locking Problem vor und das System ist nicht überlastet.
• Die Zeitaufteilung zwischen «Execute» und «Fetch» kann nicht beeinflusst werden. Dies hängt vom jeweiligen Befehl ab. Deshalb sollte nur
das Total der beiden beachtet werden.
• TKPROF zeigt grundsätzlich die Arbeit an, welche wirklich gemacht
wurde. Mit einem Vergleich zu AutoTrace, wo nur eine Schätzung vorhanden ist, kann sehr schnell gesehen werden, wo der CBO falsche
Annahmen getroffen hat, welche zu schlechter Performance führen.
• Das Verständnis des Execution Plan ist von grundlegender Bedeutung
um nun Massnahmen zur Verbesserung einzuleiten. Solche Massnahmen bestehen einerseits aus dem Anbringen von Hints, dem Legen von
Indexen oder dem Umschreiben von SQL-Statements.
Seite 25
von-25
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
2.7
Optimierung
Nach der Auswertung des Execution Plans stellt sich nun die Frage, wie
ein SQL-Statement optimiert werden kann. In erster Linie sollte immer
überlegt werden, ob das SQL-Statement optimal formuliert ist und ob die
allenfalls notwendigen Indexe vorhanden sind. Man bedenke also nochmals, dass nur 3 - 5 % der SQL-Statements Performance Probleme
aufweisen dürfen, ansonsten muss das Datenmodell und die gesamte
Konfiguration näher untersucht werden. Oracle stellt für das SQL-Tuning
sogenannte «Hints» zur Verfügung, die den Optimizer anweisen, einen
anderen Execution Plan zu wählen. Dabei handelt es sich lediglich um
einen Hint (Hinweise), und nicht um einen Befehl. Der Optimizer kann
diesen Hint also auch ignorieren. Die Angabe des Hint gehört nicht zur
Syntax des SQL-Statements, das heisst, Syntax-Fehler bei der Angabe
des Hints werden vom Optimizer nie beanstandet. Hints werden in Form
von Kommentaren dem SQL-Statement mitgegeben. Ob der Hint berücksichtigt wurde sieht man erst im Execution Plan. Es macht deshalb keinen
Sinn einen Hint anzubringen wenn nicht gleichzeitig auch der Execution
Plan dazu kontrolliert wird.
Anbringen eines Hints
Ein Hint wird dem SQL-Befehl als Kommentar und einem «+» Zeichen
mitgegeben. Er ist für den Optimizer nicht zwingend.
SELECT
FROM
WHERE
AND
Strategie zur
Optimierung
/*+ RULE */ e.ename, e.sal, d.dname
emp e, dept d
d.deptno = e.deptno
d.dname = ’SALES’;
Um rasch und gezielt eine Performance Verbesserung zu erreichen, empfiehlt sich das folgende Vorgehen.
1. Test des SQL-Statements mit dem anderen Optimizer. Vergleich des
Execution Plans der beiden Optimizer. Gleiche Execution Plans deuten
auf eine gute Optimierung hin.
COST Based Optimizer:
RULE Based Optimizer:
Hint = CHOOSE
Hint = RULE
2. Soll möglichst rasch die erste Row angezeigt werden oder ist die gesamte Verarbeitung (Batch) von Wichtigkeit ?
Möglichst rasch 1. Row anzeigen: Hint = First_Rows
Gesamtdurchsatz optimieren:
Hint = All_Rows
3. Kann der Zugriff mit einem Hint geändert werden ?
4. Kann ein Index verwendet werden ?
5. Kann das Statement umformuliert werden ?
6. Haben Sie die Regeln für gutes SQL eingehalten ?
Seite
26 von
82
Ausgabe
5. Januar
1995
Seite -26
$NDGLD $*
Oracle Datenbank Tuning
Hints
Folgende Hints stehen zur Verfügung
Hint
Erläuterung
RULE
CHOOSE
FIRST_ROWS
Verwendung des Rule Based Optimizer
Verwendung des Cost Based Optimizer
CBO Optimierung der Zeit zur Anzeige der ersten Row. IndexScans und Nested Loops werden vom Optimizer bevorzugt eingesetzt.
CBO Optimierung des Gesamtdurchsatzes. Fulltable-Scans werden vom Optimizer bevorzugt eingesetzt.
Fulltable-Scan auf angegebener Tabelle durchführen
Join der Tabellen erfolgt in der gleichen Reihenfolge wie sie in
der FROM Klausel erscheinen. Dieser Hint steht selten allein,
sondern wird vielfach in Kombination mit einem anderen Hint
angebracht.
Joine die angegebene Tabelle zum Resultat der vorangegangenen Row-Source mittels einem Sort-Merge Join.
Joine die angegebene Tabelle zum Resultat der vorangegangenen Row-Source mittels einem Hash Join.
Joine die angegebene Tabelle zum Resultat der vorangegangenen Row-Source mittels einem Nested Loop. Die angegebene
Tabelle wird im inneren Loop verarbeitet.
Verwendung des Index einer bestimmten Tabelle
Verwendung eines Index Merge auf bestimmten einzelnen
Indizes der angegebenen Tabelle.
ALL_ROWS
FULL(Table)
ORDERED
USE_MERGE(Table)
USE_HASH(Table)
USE_NL(Table)
INDEX(Table Index)
AND_EQUAL(Table
Index Index Index ...)
SELECT
FROM
WHERE
AND
HASH_AJ
ename, sal
emp
deptno = 10
ename = ’KING’;
CBO: Anti Join als Hash auflösen.
SELECT ...
WHERE deptno NOT IN (SELECT ...);
MERGE_AJ
NOT IN Abfragen möglichst vermeiden !
CBO: Anti Join als Merge -Join auflösen.
SELECT ...
WHERE deptno NOT IN (SELECT ...);
CACHE
PARALLEL
ROWID(Table)
CLUSTER(Table)
HASH (Tabelle)
NOT IN Abfragen möglichst vermeiden !
Full Table Scans lassen die Blocks solange wie möglich in der
SGA stehen. Ideal für kleine Lookup Tabellen.
Im Zusammenhang mit der Parallel Query Option einsetzbar.
Angabe der concurrent Query Server zur Verarbeitung angeben.
Table Scan über die ROWID der Tabelle.
Verwendung des Index Clusters der Tabelle.
Verwendung des Hash Clusters der Tabelle.
Die zur Verfügung stehenden Hints sind nicht für alle Oracle Versionen
verfügbar. Man beachte also nochmals, dass der Optimizer einen unbekannten Hint ignoriert und keine Fehlermeldung generiert.
Ausgabe 5. Januar 1995
Seite 27
von-27
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
2.8
Regeln für gutes SQL
Beachten Sie auch die Regeln für den Indexzugriff
(siehe Verwendung des Index auf Seite 10).
Allgemeines
• Die SELECT Liste steht zur freien Wahl. Alle Funktionen (auch Inline
Funktionen) und Möglichkeiten stehen offen.
• Die FROM Klausel soll die Tabellen in der Reihenfolge auflisten, wie
der Optimizer sie absuchen soll.
• In der WHERE Klausel sollten keine indexierten Attribute durch Funktionen / Berechnungen verändert werden, da dies den Indexzugriff verhindert.
• Die WHERE Klausel führt zuerst die Join-Attribute, danach die Attribute
gemäss ihrer Selektivität auf.
• ORDER BY Klauseln können nur auf unveränderte, indexierte NOT
NULL Attribute optimiert werden.
• Joins sind einfacher optimierbar als Subqueries.
• NOT IN sollte grundsätzlich vermieden werden. Im Zusammenhang mit
dem RULE Based Optimizer ist die Verwendung verboten !
• Komplexe Views in Joins oder Views auf Views sollten immer mit
TKPROF kontrolliert werden.
• Tabellen Aliasnamen in Joins bei allen Columns angeben, auch wenn
dies für die Eindeutigkeit nicht notwendig ist.
Outer Joins
Bei Outer Joins kann sich der Optimizer allen verfügbaren Join-Methoden
bedienen. Die Driving Table ist einfach bestimmbar, es ist die Gegenseite
zum (+), da diese Tabelle ja immer ganz gelesen werden soll.
SELECT d.dname, e.ename
FROM emp e, dept d
WHERE d.deptno = e.deptno (+);
Execution Plan
--------------------------------------------------------SELECT STATEMENT Optimizer=CHOOSE (Cost=5 Card=14
Bytes=756)
NESTED LOOPS (OUTER) (Cost=5 Card=14 Bytes=756)
TABLE ACCESS (FULL) OF ’DEPT’ (Cost=1 Card=4 Bytes=116)
TABLE ACCESS (FULL) OF ’EMP’
DISTINCT
DISTINCT im Zusammenhang mit Joins sind «Performance Killer». Das
Statement kann praktisch immer umgeschrieben werden:
Schlecht ist:
SELECT DISTINCT d.deptno, d.dname
FROM dept d, emp e
WHERE d.deptno = e.deptno;
besser ist:
SELECT d.deptno, d.dname
FROM dept d
WHERE EXISTS (SELECT ’X’
FROM emp e
WHERE e.deptno = d.deptno);
Seite
28 von
82
Ausgabe
5. Januar
1995
Seite -28
$NDGLD $*
Oracle Datenbank Tuning
NOT IN
Vermeiden Sie wenn immer möglich den Operator NOT IN
Schlecht ist:
SELECT *
FROM emp
WHERE deptno NOT IN (SELECT deptno
FROM dept
WHERE loc = ’BOSTON’);
besser ist:
SELECT *
FROM emp e
WHERE NOT EXISTS (SELECT ’X’
FROM dept d
WHERE d.deptno = e.deptno
AND d.loc = ’BOSTON’);
Inline Functions
Inline Funktionen sind eigene PL/SQL Funktionen, die sich exakt gleich
verhalten wie die bekannten Oracle Funktionen AVG, COUNT, SUM, etc.
Sie können direkt in SQL-Statements verwendet werden und erweiteren
so den Funktionsumfang der SQL Sprache mit eigenen Funktionen. Solche Funktionen (Stored Functions) sind bereits vorkompiliert und
bezüglich Performance sehr ideal !
Beispiel:
Definieren der Funktion:
CREATE OR REPLACE FUNCTION sal_sum (p_deptno IN NUMBER)
RETURN NUMBER IS
l_sum
NUMBER;
BEGIN
SELECT SUM(sal) INTO l_sum
FROM emp
WHERE deptno = p_deptno;
RETURN (l_sum);
END;
Verwenden der Funktion in normalem SQL-Statement:
SELECT deptno, dname, sal_sum(deptno) FROM dept;
Ausgabe 5. Januar 1995
Seite 29
von-29
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
3.
Tuning ausgewählter SQL-Statements
3.1
COST- oder RULE-based Optimizer ?
COST-based
Für den COST-based Optimizer ist der Entscheid zwischen FulltableScan oder Index-Scan schwierig zu fällen, wenn das Resultset gross ist.
SELECT ORIGIN_ID, TO_CHAR(CREATE_DATE,’YYYYMMDDHH24MISS’),
TO_CHAR(SYSDATE, ’YYYYMMDDHH24MISS’), MS_ID, RELOAD_ID
FROM CREDIT
WHERE (STATUS = 0 AND ACCOUNT_DATE < (SYSDATE - (1 / 24)));
call
count
cpu
elapsed
disk
query
current
rows
------- ------ -------- ---------- ---------- ---------- ---------- ---------Parse
1
0.00
0.00
0
0
0
0
Execute
1
0.00
0.00
0
0
0
0
Fetch
9792
19.88
20.43
6
12117
2
146876
------- ------ -------- ---------- ---------- ---------- ---------- ---------total
9794
19.88
20.43
6
12117
2
146876
Misses in library cache during parse: 0
Optimizer goal: CHOOSE
Parsing user id: 10 (SIPP)
Rows
Execution Plan
------- --------------------------------------------------0 SELECT STATEMENT GOAL: CHOOSE
147334 TABLE ACCESS GOAL: ANALYZED (FULL) OF ’CREDIT’
********************************************************************************
RULE-based
Wie verhält sich der RULE-based Optimizer ?
SELECT /*+ RULE */ ORIGIN_ID, TO_CHAR(CREATE_DATE,’YYYYMMDDHH24MISS’),
TO_CHAR(SYSDATE, ’YYYYMMDDHH24MISS’), MS_ID, RELOAD_ID
FROM CREDIT
WHERE (STATUS = 0 AND ACCOUNT_DATE < (SYSDATE - (1 / 24)))
call
count
cpu
elapsed
disk
query
current
rows
------- ------ -------- ---------- ---------- ---------- ---------- ---------Parse
1
0.01
0.00
0
0
0
0
Execute
1
0.00
0.00
0
0
0
0
Fetch
9806
20.55
22.50
2251
12142
2
147075
------- ------ -------- ---------- ---------- ---------- ---------- ---------total
9808
20.56
22.50
2251
12142
2
147075
Misses in library cache during parse: 1
Optimizer goal: RULE
Parsing user id: 10 (SIPP)
Rows
Execution Plan
------- --------------------------------------------------0 SELECT STATEMENT GOAL: HINT: RULE
147626 TABLE ACCESS GOAL: ANALYZED (FULL) OF ’CREDIT’
********************************************************************************
Der RULE-based Optimizer versucht den Zugriff wenn möglich über
einen Index herzustellen. Da hier jedoch kein Index vorhanden ist, wählt
auch er einen Full-Table Scan. Der Bitmapped Index auf dem Attribut
STATUS kann nur vom COST-based Optimizer genutzt werden.
Seite
30 von
82
Ausgabe
5. Januar
1995
Seite -30
$NDGLD $*
Oracle Datenbank Tuning
Index-Zugriff
Nun ermöglichen wir einen Indexzugriff:
B-Tree Index auf ACCOUNT_DATE, Bitmapped Index auf STATUS:
CREATE INDEX CREDIT_ACCOUNT_DATE_IDX ON CREDIT (ACCOUNT_DATE)
TABLESPACE IDX_SIPP STORAGE (INITIAL 1M NEXT 1M
MAXEXTENTS UNLIMITED PCTINCREASE 0);
CREATE BITMAP INDEX CREDIT_STATUS_BMIDX ON CREDIT (STATUS)
TABLESPACE IDX_SIPP STORAGE (INITIAL 1M NEXT 1M
MAXEXTENTS UNLIMITED PCTINCREASE 0);
SELECT ORIGIN_ID, TO_CHAR(CREATE_DATE,’YYYYMMDDHH24MISS’),
TO_CHAR(SYSDATE, ’YYYYMMDDHH24MISS’), MS_ID, RELOAD_ID
FROM CREDIT
WHERE (STATUS = 0 AND ACCOUNT_DATE < (SYSDATE - (1 / 24)))
call
count
cpu
elapsed
disk
query
current
rows
------- ------ -------- ---------- ---------- ---------- ---------- ---------Parse
1
0.01
0.01
0
0
0
0
Execute
1
0.00
0.00
0
0
0
0
Fetch
9792
28.10
30.12
798
294688
0
146878
------- ------ -------- ---------- ---------- ---------- ---------- ---------total
9794
28.11
30.13
798
294688
0
146878
Misses in library cache during parse: 1
Optimizer goal: CHOOSE
Parsing user id: 10 (SIPP)
Rows
Execution Plan
------- --------------------------------------------------0 SELECT STATEMENT GOAL: CHOOSE
147329 TABLE ACCESS GOAL: ANALYZED (BY ROWID) OF ’CREDIT’
147330
INDEX (RANGE SCAN) OF ’CREDIT_ACCOUNT_DATE_IDX’ (NON-UNIQUE)
================================================================================
147’329
TABLE ACCESS (BY ROWID)
(CREDIT)
147’334
INDEX (RANGE SCAN)
(CREDIT_ACCOUNT_DATE_IDX
Da sehr viele Rows zurückkommmen ist der Full-Table Scan deutlich
schneller. Index auf ACCOUNT_DATE wieder droppen !
Ausgabe 5. Januar 1995
Seite 31
von-31
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
3.2
Join-Methoden (Algorithmen)
Join-Methoden
In diesem Beispiel wird die Column MSISDN selektiert, das Result-Set
besteht aus genau einer Row.
SELECT
FROM
WHERE
AND
AND
AND
AND
MSISDN
CUSTOMER, MSISDN, BINDING, SIM
CUSTOMER.CUS_ID = MSISDN.CUS_ID
MSISDN.MS_ID = BINDING.MS_ID
BINDING.IMSI = SIM.IMSI
BINDING.STATUS = 0
SRN BETWEEN ’8941019622417348629’ AND ’89410196224173486299’
call
count
cpu
elapsed
disk
query
current
rows
------- ------ -------- ---------- ---------- ---------- ---------- ---------Parse
1
0.00
0.00
0
0
0
0
Execute
2
0.00
0.00
0
0
0
0
Fetch
1
2.71
3.41
7
3238
4
1
------- ------ -------- ---------- ---------- ---------- ---------- ---------total
4
2.71
3.41
7
3238
4
1
Misses in library cache during parse: 0
Optimizer goal: CHOOSE
Parsing user id: 10 (SIPP)
Rows
Execution Plan
------- --------------------------------------------------0 SELECT STATEMENT GOAL: CHOOSE
1 NESTED LOOPS
2
HASH JOIN
101995
TABLE ACCESS GOAL: ANALYZED (FULL) OF ’MSISDN’
1
NESTED LOOPS
101995
TABLE ACCESS GOAL: ANALYZED (FULL) OF ’SIM’
1
TABLE ACCESS GOAL: ANALYZED (BY ROWID) OF ’BINDING’
2
INDEX GOAL: ANALYZED (RANGE SCAN) OF ’BINDING_IDX1’ (UNIQUE)
1
INDEX GOAL: ANALYZED (UNIQUE SCAN) OF ’PK_CUSTOMER’ (UNIQUE)
********************************************************************************
Seite
32 von
82
Ausgabe
5. Januar
1995
Seite -32
$NDGLD $*
Oracle Datenbank Tuning
Execution Plan grafisch dargestellt:
1
NL
INDEX (UNQ SCAN)
(PK_CUSTOMER)
2
HASH
JOIN
1
101’995
NL
101’995
TABLE ACCESS (FULL)
(MSISDN)
INDEX (RANGE SCAN)
(BINDING_IDX1)
TABLE ACCESS (BY ROWID)
(BINDING)
TABLE ACCESS (FULL)
(SIM)
Der Join zwischen der Tabelle SIM und BINDING erfolgt in einem Nested
Loop. Dazu wird die gesamte Tabelle SIM gelesen und der innere Loop
wird 101’995 mal ausgeführt um genau eine Row zurückzubringen. Der
Join zwischen diesem Ergebnis (SIM, BINDING) und der Tabelle
MSISDN wird in einem HASH-Join vorgenommen, die Tabelle MSISDN
wird ganz gelesen, der Join bringt 2 Rows zurück. Der Join zwischen diesem Ergebnis und der Tabelle CUSTOMER wird wieder in einem Nested
Loop durchgeführt, der innere Loop wird nur noch 2 mal ausgeführt, über
den Primary Key der Tabelle CUSTOMER wird das Result-Set von einer
Row schliesslich zurückgegeben. Der erste NL ist hier schlecht, da er
101’995 mal ausgeführt werden muss.
Ausgabe 5. Januar 1995
Seite 33
von-33
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
Da wir wissen, dass die Tabelle SIM einen UNIQUE Index auf dem Attribut SRN hat, geben wir dies dem Optimizer in einem Hint bekannt. Da nur
eine einzige Row zurückkommt ist es von Vorteil alle Joins über einen
Nested Loop Algorithmus zu joinen, dazu dient der Hint FIRST_ROWS.
SELECT
FROM
WHERE
AND
AND
AND
AND
/*+ INDEX(SIM SIM_IDX1) FIRST_ROWS */ MSISDN
CUSTOMER, MSISDN, BINDING, SIM
CUSTOMER.CUS_ID = MSISDN.CUS_ID
MSISDN.MS_ID = BINDING.MS_ID
BINDING.IMSI = SIM.IMSI
BINDING.STATUS = 0
SRN BETWEEN ’8941019622417348629’ AND ’89410196224173486299’;
call
count
cpu
elapsed
disk
query
current
rows
------- ------ -------- ---------- ---------- ---------- ---------- ---------Parse
2
0.00
0.00
0
0
0
0
Execute
2
0.00
0.00
0
0
0
0
Fetch
2
0.00
0.00
0
32
0
2
------- ------ -------- ---------- ---------- ---------- ---------- ---------total
6
0.00
0.00
0
32
0
2
Misses in library cache during parse: 0
Optimizer goal: FIRST_ROWS
Parsing user id: 10 (SIPP)
Rows
Execution Plan
------- --------------------------------------------------0 SELECT STATEMENT GOAL: HINT: FIRST_ROWS
1 NESTED LOOPS
1
NESTED LOOPS
1
NESTED LOOPS
1
TABLE ACCESS GOAL: ANALYZED (BY ROWID) OF ’SIM’
2
INDEX GOAL: ANALYZED (RANGE SCAN) OF ’SIM_IDX1’ (UNIQUE)
1
TABLE ACCESS GOAL: ANALYZED (BY ROWID) OF ’BINDING’
2
INDEX GOAL: ANALYZED (RANGE SCAN) OF ’BINDING_IDX1’ (UNIQUE)
1
TABLE ACCESS GOAL: ANALYZED (BY ROWID) OF ’MSISDN’
1
INDEX GOAL: ANALYZED (UNIQUE SCAN) OF ’PK_MSISDN’ (UNIQUE)
1
INDEX GOAL: ANALYZED (UNIQUE SCAN) OF ’PK_CUSTOMER’ (UNIQUE)
********************************************************************************
Execution Plan grafisch dargestellt:
1
NL
1
INDEX (UNIQUE SCAN)
(PK_CUSTOMER)
NL
1
INDEX (UNIUQ SCAN)
(PK_MSISDN)
TABLE ACCESS (BY ROWID)
(MSISDN)
INDEX (RANGE SCAN)
(BINDING_IDX1)
TABLE ACCESS (BY ROWID)
(BINDING)
NL
1
TABLE ACCESS (BY ROWID)
(SIM)
2
INDEX (RANGE SCAN)
(SIM_IDX1)
Auf jeder Stufe wird nur noch das Notwendigste verarbeitet, die äusseren
Loops werden nie mehr als 2 mal durchlaufen, dementsprechend schnell
wird die gesuchte MSISDN gefunden.
Seite
34 von
82
Ausgabe
5. Januar
1995
Seite -34
$NDGLD $*
Oracle Datenbank Tuning
Wie verhält sich in diesem Fall der RULE based Optimizer ?
SELECT
FROM
WHERE
AND
AND
AND
AND
/*+ RULE */ MSISDN
CUSTOMER, MSISDN, BINDING, SIM
CUSTOMER.CUS_ID = MSISDN.CUS_ID
MSISDN.MS_ID = BINDING.MS_ID
BINDING.IMSI = SIM.IMSI
BINDING.STATUS = 0
SRN BETWEEN ’8941019622417348629’ AND ’89410196224173486299’
call
count
cpu
elapsed
disk
query
current
rows
------- ------ -------- ---------- ---------- ---------- ---------- ---------Parse
1
0.00
0.00
0
0
0
0
Execute
1
0.00
0.00
0
0
0
0
Fetch
1
0.00
0.00
0
16
0
1
------- ------ -------- ---------- ---------- ---------- ---------- ---------total
3
0.00
0.00
0
16
0
1
Misses in library cache during parse: 0
Optimizer goal: RULE
Parsing user id: 10 (SIPP)
Rows
Execution Plan
------- --------------------------------------------------0 SELECT STATEMENT GOAL: HINT: RULE
1 NESTED LOOPS
1
NESTED LOOPS
1
NESTED LOOPS
1
TABLE ACCESS GOAL: ANALYZED (BY ROWID) OF ’SIM’
2
INDEX GOAL: ANALYZED (RANGE SCAN) OF ’SIM_IDX1’ (UNIQUE)
1
TABLE ACCESS GOAL: ANALYZED (BY ROWID) OF ’BINDING’
2
INDEX GOAL: ANALYZED (RANGE SCAN) OF ’BINDING_IDX1’ (UNIQUE)
1
TABLE ACCESS GOAL: ANALYZED (BY ROWID) OF ’MSISDN’
1
INDEX GOAL: ANALYZED (UNIQUE SCAN) OF ’PK_MSISDN’ (UNIQUE)
1
INDEX GOAL: ANALYZED (UNIQUE SCAN) OF ’PK_CUSTOMER’ (UNIQUE)
********************************************************************************
Der RULE based Optimizer wählt in diesem Fall den optimalen Execution
Plan, da er wenn möglich Indexe verwendet und Fulltable Scans vermeidet. Die Lösung besteht hier also den RULE based Optimizer zu
verwenden, da dadurch der Hint einfacher wird.
Ausgabe 5. Januar 1995
Seite 35
von-35
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
3.3
Ein oder mehrere Attribute in der Selektion
SELECT * ...
Es stellt sich nun die Frage, ob sich der Execution Plan ändert, wenn alle
Attribute in die Selektion aufgenommen werden.
SELECT
FROM
WHERE
AND
AND
AND
AND
*
CUSTOMER, MSISDN, BINDING, SIM
CUSTOMER.CUS_ID = MSISDN.CUS_ID
MSISDN.MS_ID = BINDING.MS_ID
BINDING.IMSI = SIM.IMSI
BINDING.STATUS = 0
SRN BETWEEN ’8941019622417348629’ AND ’89410196224173486299’
call
count
cpu
elapsed
disk
query
current
rows
------- ------ -------- ---------- ---------- ---------- ---------- ---------Parse
1
0.00
0.00
0
0
0
0
Execute
2
0.01
0.01
0
0
0
0
Fetch
1
9.80
14.62
5
4075
8
1
------- ------ -------- ---------- ---------- ---------- ---------- ---------total
4
9.81
14.63
5
4075
8
1
Misses in library cache during parse: 0
Optimizer goal: CHOOSE
Parsing user id: 10 (SIPP)
Rows
Execution Plan
------- --------------------------------------------------0 SELECT STATEMENT GOAL: CHOOSE
1 HASH JOIN
101995
TABLE ACCESS GOAL: ANALYZED (FULL) OF ’SIM’
203888
HASH JOIN
1940
TABLE ACCESS GOAL: ANALYZED (FULL) OF ’CUSTOMER’
181250
HASH JOIN
101998
TABLE ACCESS GOAL: ANALYZED (FULL) OF ’BINDING’
101995
TABLE ACCESS GOAL: ANALYZED (FULL) OF ’MSISDN’
********************************************************************************
1
HASH
JOIN
101’995
203’888
TABLE ACCESS (FULL)
(SIM)
HASH
JOIN
181’250
1’940
HASH
JOIN
101’995
TABLE ACCESS (FULL)
(MSISDN)
TABLE ACCESS (FULL)
(CUSTOMER)
101’998
TABLE ACCESS (FULL)
(BINDING)
In diesem Fall tritt das typische Problem des COST based Optimizers
zutage, er schätzt das zu erwartende Result Set falsch ein. Er wählt für
jeden Join eine Hash-Algorithmus, der für ein grosses Result Set geeignet ist. In diesem Fall wird aber nur eine Row selektiert,
dementsprechend ist der gewählte Execution Plan in diesem Fall sehr
ungeeignet. Wie verhält sich der RULE Based Optimizer ?
Seite
36 von
82
Ausgabe
5. Januar
1995
Seite -36
$NDGLD $*
Oracle Datenbank Tuning
Wie verhält sich in diesem Fall der RULE based Optimizer ?
SELECT
FROM
WHERE
AND
AND
AND
AND
/*+ RULE */ *
CUSTOMER, MSISDN, BINDING, SIM
CUSTOMER.CUS_ID = MSISDN.CUS_ID
MSISDN.MS_ID = BINDING.MS_ID
BINDING.IMSI = SIM.IMSI
BINDING.STATUS = 0
SRN BETWEEN ’8941019622417348629’ AND ’89410196224173486299’
call
count
cpu
elapsed
disk
query
current
rows
------- ------ -------- ---------- ---------- ---------- ---------- ---------Parse
1
0.01
0.02
0
0
0
0
Execute
1
0.00
0.00
0
0
0
0
Fetch
1
0.00
0.00
0
17
0
1
------- ------ -------- ---------- ---------- ---------- ---------- ---------total
3
0.01
0.02
0
17
0
1
Misses in library cache during parse: 1
Optimizer goal: RULE
Parsing user id: 10 (SIPP)
Rows
Execution Plan
------- --------------------------------------------------0 SELECT STATEMENT GOAL: HINT: RULE
1 NESTED LOOPS
1
NESTED LOOPS
1
NESTED LOOPS
1
TABLE ACCESS GOAL: ANALYZED (BY ROWID) OF ’SIM’
2
INDEX GOAL: ANALYZED (RANGE SCAN) OF ’SIM_IDX1’ (UNIQUE)
1
TABLE ACCESS GOAL: ANALYZED (BY ROWID) OF ’BINDING’
2
INDEX GOAL: ANALYZED (RANGE SCAN) OF ’BINDING_IDX1’ (UNIQUE)
1
TABLE ACCESS GOAL: ANALYZED (BY ROWID) OF ’MSISDN’
1
INDEX GOAL: ANALYZED (UNIQUE SCAN) OF ’PK_MSISDN’ (UNIQUE)
1
TABLE ACCESS GOAL: ANALYZED (BY ROWID) OF ’CUSTOMER’
1
INDEX GOAL: ANALYZED (UNIQUE SCAN) OF ’PK_CUSTOMER’ (UNIQUE)
********************************************************************************
Der RULE based Optimizer wählt hier den optimalsten Execution Plan.
Ausgabe 5. Januar 1995
Seite 37
von-37
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
3.4
Subquery Optimierung
Sehr oft finden Vergleiche zwischen Tabellen statt, im folgenden Beispiel
wird untersucht, welche Teilnehmer (MSISDN) noch nie einen Reload
(CREDIT) durchgeführt haben. Man untersucht also, welche MSISDN
Nummern nicht in der Tabelle CREDIT vorhanden sind.
Variante 1
Mittels MINUS
select count(*)
from msisdn
where ms_id in (select ms_id from msisdn minus select ms_id from credit)
call
count
cpu
elapsed
disk
query
current
rows
------- ------ -------- ---------- ---------- ---------- ---------- ---------Parse
2
0.01
0.02
0
0
0
0
Execute
4
0.01
0.15
0
0
0
0
Fetch
2
24.19
33.06
4974
8365
2246
2
------- ------ -------- ---------- ---------- ---------- ---------- ---------total
8
24.21
33.23
4974
8365
2246
2
Misses in library cache during parse: 1
Optimizer goal: CHOOSE
Parsing user id: 10 (SIPP)
Rows
Execution Plan
------- --------------------------------------------------0 SELECT STATEMENT GOAL: CHOOSE
0 SORT (AGGREGATE)
95087
HASH JOIN
101995
TABLE ACCESS GOAL: ANALYZED (FULL) OF ’MSISDN’
64166
VIEW
139824
MINUS
101995
SORT (UNIQUE)
101995
TABLE ACCESS GOAL: ANALYZED (FULL) OF ’MSISDN’
156788
SORT (UNIQUE)
156788
TABLE ACCESS GOAL: ANALYZED (FULL) OF ’CREDIT’
********************************************************************************
Variante 2
Mittels NOT IN
select count(*)
from msisdn
where ms_id not in (select ms_id from credit)
call
count
cpu
elapsed
disk
query
current
rows
------- ------ -------- ---------- ---------- ---------- ---------- ---------Parse
2
0.00
0.01
0
0
0
0
Execute
2
0.00
0.00
0
0
0
0
Fetch
2
24.72
26.69
283
818552
4
2
------- ------ -------- ---------- ---------- ---------- ---------- ---------total
6
24.72
26.70
283
818552
4
2
Misses in library cache during parse: 0
Optimizer goal: CHOOSE
Parsing user id: 10 (SIPP)
Rows
Execution Plan
------- --------------------------------------------------0 SELECT STATEMENT GOAL: CHOOSE
0 SORT (AGGREGATE)
101995
FILTER
101995
TABLE ACCESS GOAL: ANALYZED (FULL) OF ’MSISDN’
101995
INDEX GOAL: ANALYZED (RANGE SCAN) OF ’CREDIT_IDX1’ (NON-UNIQUE)
********************************************************************************
Seite
38 von
82
Ausgabe
5. Januar
1995
Seite -38
$NDGLD $*
Oracle Datenbank Tuning
Variante 3
Mittels NOT EXISTS
select count(*)
from msisdn m
where not exists (select ’x’ from credit c where c.ms_id = m.ms_id)
call
count
cpu
elapsed
disk
query
current
rows
------- ------ -------- ---------- ---------- ---------- ---------- ---------Parse
2
0.00
0.00
0
0
0
0
Execute
2
0.00
0.00
0
0
0
0
Fetch
2
24.76
27.23
290
818552
4
2
------- ------ -------- ---------- ---------- ---------- ---------- ---------total
6
24.76
27.23
290
818552
4
2
Misses in library cache during parse: 0
Optimizer goal: CHOOSE
Parsing user id: 10 (SIPP)
Rows
Execution Plan
------- --------------------------------------------------0 SELECT STATEMENT GOAL: CHOOSE
0 SORT (AGGREGATE)
101995
FILTER
101995
TABLE ACCESS GOAL: ANALYZED (FULL) OF ’MSISDN’
101995
INDEX GOAL: ANALYZED (RANGE SCAN) OF ’CREDIT_IDX1’ (NON-UNIQUE)
********************************************************************************
Variante 4
Mittels Outer Join
select count(*)
from msisdn m, credit c
where m.ms_id = c.ms_id (+)
and c.ms_id is null
call
count
cpu
elapsed
disk
query
current
rows
------- ------ -------- ---------- ---------- ---------- ---------- ---------Parse
2
0.00
0.00
0
0
0
0
Execute
4
0.00
0.00
0
0
0
0
Fetch
2
11.34
21.50
4659
6656
8
2
------- ------ -------- ---------- ---------- ---------- ---------- ---------total
8
11.34
21.50
4659
6656
8
2
Misses in library cache during parse: 0
Optimizer goal: CHOOSE
Parsing user id: 10 (SIPP)
Rows
Execution Plan
------- --------------------------------------------------0 SELECT STATEMENT GOAL: CHOOSE
0 SORT (AGGREGATE)
220956
FILTER
231345
HASH JOIN (OUTER)
101995
TABLE ACCESS GOAL: ANALYZED (FULL) OF ’MSISDN’
156790
TABLE ACCESS GOAL: ANALYZED (FULL) OF ’CREDIT’
Die Variante 2 darf mit dem RULE-based Optimizer auf keinen Fall angewendet werden. NOT IN ist grundsätzlich mit dem RULE-based Optimizer
verboten. Der COST-based Optimizer erkennt das NOT IN und wandelt
es in ein EXISTS um. Interessant ist die Variante 4 (Outer Join), jedoch
schwer verständlich und sollte eigentlich aus diesem Grund vermieden
werden.
Ausgabe 5. Januar 1995
Seite 39
von-39
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
3.5
Zählen von Rows
COUNT
Zählen der Anzahl Rows (1’537’723) in der sehr grosse Tabelle CDR.
select count(*)
from
cdr;
call
count
cpu
elapsed
disk
query
current
rows
------- ------ -------- ---------- ---------- ---------- ---------- ---------Parse
3
0.00
0.00
0
0
0
0
Execute
3
0.00
0.00
0
0
0
0
Fetch
3
50.16
144.41
191775
192321
9
3
------- ------ -------- ---------- ---------- ---------- ---------- ---------total
9
50.16
144.41
191775
192321
9
3
Misses in library cache during parse: 0
Optimizer goal: CHOOSE
Parsing user id: 10 (SIPP)
Rows
Execution Plan
------- --------------------------------------------------0 SELECT STATEMENT GOAL: CHOOSE
0 SORT (AGGREGATE)
1537723
TABLE ACCESS GOAL: ANALYZED (FULL) OF ’CDR’
********************************************************************************
Der COST-based Optimizer wählt einen Full-Table Scan, der Aufwand ist
beträchtlich. Wie verhält es sich wenn man einen Index-Scan auf dem
Primary Key wählt ?
select /*+ index (cdr pk_cdr) */ count(*)
from
cdr;
call
count
cpu
elapsed
disk
query
current
rows
------- ------ -------- ---------- ---------- ---------- ---------- ---------Parse
4
0.02
0.01
0
0
0
0
Execute
4
0.00
0.00
0
0
0
0
Fetch
4
19.57
41.89
8746
42640
0
4
------- ------ -------- ---------- ---------- ---------- ---------- ---------total
12
19.59
41.90
8746
42640
0
4
Misses in library cache during parse: 0
Optimizer goal: CHOOSE
Parsing user id: 10 (SIPP)
Rows
Execution Plan
------- --------------------------------------------------0 SELECT STATEMENT GOAL: CHOOSE
0 SORT (AGGREGATE)
1537723
INDEX GOAL: ANALYZED (FULL SCAN) OF ’PK_CDR’ (UNIQUE)
********************************************************************************
Der Aufwand sinkt beträchtlich, da hier nur der gesamte Index sequentiell
auf dem Primary Key gelesen werden muss.
Seite
40 von
82
Ausgabe
5. Januar
1995
Seite -40
$NDGLD $*
Oracle Datenbank Tuning
3.6
SQL-Funktionen die Indexzugriff verhindern
Jede SQL Funktion auf der linken Seite der WHERE Klausel verhindert
einen Indexzugriff. Oft kann das Statement umformuliert werden um den
Indexzugriff zu ermöglichen wie das folgende Beispiel zeigt:
SELECT COUNT(*)
FROM acm
WHERE TRUNC(date_cdr1) = TRUNC(sysdate);
call
count
cpu
elapsed
disk
query
current
rows
------- ------ -------- ---------- ---------- ---------- ---------- ---------Parse
1
0.00
0.01
0
0
0
0
Execute
1
0.00
0.00
0
0
0
0
Fetch
1
3.50
4.83
309
1379
2
1
------- ------ -------- ---------- ---------- ---------- ---------- ---------total
3
3.50
4.84
309
1379
2
1
Misses in library cache during parse: 1
Optimizer goal: CHOOSE
Parsing user id: 10 (SIPP)
Rows
Execution Plan
------- --------------------------------------------------0 SELECT STATEMENT GOAL: CHOOSE
0 SORT (AGGREGATE)
126994
TABLE ACCESS GOAL: ANALYZED (FULL) OF ’ACM’
********************************************************************************
Mit einem Index auf date_cdr1 kann die Abfrgae sehr beschleunigt
werden, aber das Statement muss umgeschrieben werden so dass der
Index auch gebraucht werden kann.
CREATE INDEX acm_date_cdr1_btidx ON SIPP.ACM (DATE_CDR1)
TABLESPACE idx_sipp
STORAGE (INITIAL 5M NEXT 5M MINEXTENTS 1 MAXEXTENTS UNLIMITED PCTINCREASE 0)
PCTFREE 10;
ANALYZE INDEX acm_date_cdr1_btidx COMPUTE STATISTICS;
SELECT COUNT(*)
FROM acm
WHERE date_cdr1 BETWEEN TRUNC(sysdate) AND TRUNC(sysdate) + .99999;
call
count
cpu
elapsed
disk
query
current
rows
------- ------ -------- ---------- ---------- ---------- ---------- ---------Parse
1
0.01
0.01
0
0
0
0
Execute
1
0.00
0.00
0
0
0
0
Fetch
1
0.00
0.00
0
3
0
1
------- ------ -------- ---------- ---------- ---------- ---------- ---------total
3
0.01
0.01
0
3
0
1
Misses in library cache during parse: 1
Optimizer goal: CHOOSE
Parsing user id: 10 (SIPP)
Rows
Execution Plan
------- --------------------------------------------------0 SELECT STATEMENT GOAL: CHOOSE
0 SORT (AGGREGATE)
74
FILTER
75
INDEX GOAL: ANALYZED (RANGE SCAN) OF ’ACM_DATE_CDR1_BTIDX’ (NONUNIQUE)
********************************************************************************
Ausgabe 5. Januar 1995
Seite 41
von-41
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
4.
Datenbank Tuning
Übersicht
Das Datenbank Tuning wird in die folgenden Teilgebiete gegliedert:
•
•
•
•
•
•
Cache und Memory Optimierung
Indexierungs Strategie
Fragmentierungen
I/O-Optimierung
Block Contention
Transaktionen und Locking
Im Performance Pack können einige der genannten Überwachungsaufgaben wahrgenommen werden. Für detailliertere Auswertungen werden
weiterhin Scripts eingesetzt.
4.1
Oracle Performance Pack
Overview Window
Seite
42 von
82
Ausgabe
5. Januar
1995
Der Performance Manager zeigt die wichtigsten Parameter im
«Overview» Window.
Seite -42
$NDGLD $*
Oracle Datenbank Tuning
Buffer Cache Hit %
Der Buffer Cache enthält Kopien der gelesenen Daten Blocks aus den
Datenbankfiles. Alle Sessions teilen sich den Buffer Cache, der Inhalt des
Buffer Caches wird gemäss einem LRU Algorithmus mittels DBWR auf
die DB geschrieben. Muss ein Block im Datenbankfile gelesen werden,
so handelt es sich dabei um einen Cache Miss, wird der Block bereits im
Memory gefunden so spricht man von einem Cache Hit.
Die Tabelle V$SYSSTAT zeigt die kumulierten Werte seit die Instance
gestartet wurde. Die Physical Reads seit dem Instance Startup verschlechtert die Hit-Ratio, nach einer bestimmten Zeit pendelt sich der
Wert ein.
Es sollte ein HitRatio mit folgenden Werten angestrebt werden:
Datenbank mit vielen Online-Transaction intensiven Benutzern: .... 98%
Batch intensive Applikation: .............................................................89%
Berechnung
HitRatio % = 1 -
SQL Statements
SUM (Physical Reads)
* 100
SUM (Consistent Gets + DB Block Gets)
Berechnung der Buffer Cache HitRatio in %
select (1 - (sum(decode(a.name,’physical reads’,value,0)) /
(sum(decode(a.name,’consistent gets’,value,0)) +
sum(decode(a.name,’db block gets’,value,0))))) * 100,
((sum(decode(a.name,’physical reads’,value,0)) /
(sum(decode(a.name,’consistent gets’,value,0)) +
sum(decode(a.name,’db block gets’,value,0))))) * 100
from v$sysstat a;
Bei einer schlechtem HitRatio sollten zuerst die Queries mit dem grössten
Einfluss auf die HitRatio isoliert werden. Anschliessend muss unter
Umständen der Wert für DB_BLOCK_BUFFERS erhöht werden.
Ausgabe 5. Januar 1995
Seite 43
von-43
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
Session mit
schlechtem Hit-Ratio
Welche Sessions verursachen den grössten negativen Effekt auf die
Instance HitRatio ?
select substr(a.username,1,12) "User",
a.sid "sid",
b.consistent_gets "ConsGets",
b.block_gets "BlockGets",
b.physical_reads "PhysReads",
100 * round((b.consistent_gets + b.block_gets - b.physical_reads) /
(b.consistent_gets + b.block_gets),3) HitRatio
from v$session a, v$sess_io b
where a.sid = b.sid
and (b.consistent_gets + b.block_gets) > 0
and a.username is not null
order by HitRatio asc;
Sessions mit einem
schlechten Buffer
Cache Ratio. Diese
Session benützen am
meisten Ressourcen.
Logical Reads
Seite
44 von
82
Ausgabe
5. Januar
Logical Reads ist die Summe der Consistent Gets + Block Gets. Sowohl
für Consistent Gets wie auch für Block Gets erfolgt kein physisches I/O,
diese Buffer werden im Memory gefunden.
1995
Seite -44
$NDGLD $*
Oracle Datenbank Tuning
Physisches I/O
Das Diagramm zeigt die Auswirkungen des physischen I/O bei einem
konstanten logischen Read (Consistent Gets + Block Gets).
100
90
80
Hit Ratio %
70
Log. Reads - Phys. Reads
60
* 100
Log. Reads
50
40
30
20
10
0
100
200
300
400
500
600
700
800
900
1000
Physical Reads
Je weniger Informationen im Buffer Cache gefunden werden um so mehr
physisches I/O wird generiert, um die Daten aus der Datenbank zu lesen.
Ausgabe 5. Januar 1995
Seite 45
von-45
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
Library Cache Hit %
Der Library Cache ist Teil des Shared Pools.
Shared Pool
Buffer Cache
Redolog Cache
Dictionary Cache
Library Cache
Shared SQL Area
PL/SQL Procedures
and Packages
Cache Misses im Library Cache sind «sehr teuer», da das SQL-Statement geladen, geparst und ausgeführt werden muss. Hier gilt die Regel,
dass 99 % aller SQL-Statements in geparster Form im Memory vorliegen
müssen. Ist dies nicht der Fall so muss der Wert SHARED_POOL_SIZE
erhöht werden.
Berechnung
HitRatio % = 1 -
SUM (Reloads)
SUM (Pins)
* 100
SQL-Statement
select (1-(sum(reloads)/sum(pins))) *100 from v$librarycache;
Data Dictionary
Cache Hit %
Der Data Dictionary Cache ist Teil des Shared Pools. Nach dem Instance
Startup werden die Data Dictionary Informationen ins Memory geladen.
Nach einer gewissen Betriebszeit sollte sich ein stabiler Zustand einstellen. Der Data Dictionary Cache Miss sollte kleiner als 10 % sein.
Berechnung
HitRatio % = 1 -
SUM (Getmisses)
SUM (Gets)
* 100
SQL-Statement
select (1- (sum(getmisses)/sum(gets))) * 100
from v$rowcache;
Memory Sort Hit %
Anteil der Sort Operationen im Memory:
Berechnung
HitRatio % =
SQL-Statement
Seite
46 von
82
Ausgabe
5. Januar
Sorts (Memory)
Sorts (Memory) + Sorts(Disk)
* 100
select (sum(decode(name, ’sorts (memory)’, value, 0)) /
(sum(decode(name, ’sorts (memory)’, value, 0)) +
sum(decode(name, ’sorts (disk)’, value, 0)))) * 100
from v$sysstat;
1995
Seite -46
$NDGLD $*
Oracle Datenbank Tuning
Rollback Nowait %
Das Diagramm Rollback Nowait Hit % zeigt die Hits und Misses für die
Online Rollback Segmente. Ist dieser Wert zu gross, so müssen mehr
Rollbacksegmente erstellt werden.
Berechnung
HitRatio % =
Sum(Gets) - Sum(Waits)
* 100
Sum(Gets)
SQL-Statement
select ((sum(gets)-sum(waits)) / sum(gets)) * 100
from v$rollstat;
Rollback Segment
Waits
Rollback Segment Waits können einfach aus v$waitstat gelesen werden.
SELECT * from v$waitstat;
CLASS
COUNT
TIME
------------------ ---------- ---------data block
78
147
sort block
0
0
save undo block
0
0
segment header
0
0
save undo header
0
0
free list
0
0
system undo header
0
0
system undo block
0
0
undo header
124
139 <---- !
undo block
13
3 <---- !
Waits auf «undo header» werden häufig verringert, indem man weitere
Rollback Segmente erstellt.
Waits auf «undo block» werden verringert, indem man Rollback Segmente mit mehr Extents erstellt (10 - 20 Extents).
Rollback Segments
Shrinks
Rollbacksegmente sollten nicht dauernd wachsen und wieder kleiner werden, um den OPTIMAL Parameter einzuhalten. Dies kann mit dem
folgenden Query kontrolliert werden. EXTENTS und SHRINKS sollten
keine auftreten, sonst muss der Parameter OPTIMAL angepasst werden.
SELECT name, extents, rssize, writes, xacts,
gets, waits, extends, shrinks, aveactive
FROM v$rollstat stat, v$rollname name
WHERE stat.usn = name.usn
AND status = ’ONLINE’;
Keine Extents und Shrinks = OK
Ausgabe 5. Januar 1995
Seite 47
von-47
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
File I/O Rate
Das File I/O Rate Diagramm zeigt die Anzahl physical reads
und writes pro Sekunde der Oracle Datenbankfiles der gesamten
Instance.
SQL-Statement
select sum(phyrds), sum(phywrts) from v$filestat;
Phys. Writes / s (blau)
Phys. Reads / s (grün)
System I/O Rate
Das System I/O Rate Diagramm zeigt die Anzahl logischen und physischen Reads sowie die Anzahl Blockänderungen pro Sekunde.
SQL-Statement
select (sum(decode(name,’db block gets’, value,0)) +
sum(decode(name,’consistent gets’, value,0))),
sum(decode(name,’db block changes’, value,0)),
sum(decode(name,’physical reads’, value,0))
from v$sysstat;
Logical Reads/s (grün)
Db Block Changes / s (blau)
Phys. Reads / s (rot)
Logical Reads = Db Block Gets + Consistent Gets
Seite
48 von
82
Ausgabe
5. Januar
1995
Seite -48
$NDGLD $*
Oracle Datenbank Tuning
Throughput
Das Diagramm zeigt die Anzahl User Calls und Transaktionen pro
Sekunde der gesamten Instance.
SQL-Statement
select sum(decode(name,’user commits’, value,0)),
sum(decode(name,’user calls’, value,0))
from v$sysstat;
User Calls/s (grün)
Transactions / s (blau)
Active Users
Das Diagramm «Active Users» zeigt die Anzahl aktive User Sessions,
welche die Datenbank Instance zur Zeit benutzen.
SQL-Statement
select
from
where
and
Ausgabe 5. Januar 1995
count(*)
v$session
status=’ACTIVE’
type!=’BACKGROUND’;
Seite 49
von-49
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
Circuit
Das Diagramm «Circuit» zeigt die virtuellen Verbindung zur Datenbank
Instance via Dispatcher und Shared Serverprozess, welcher den User
Process verarbeitet.
SQL-Statement
select rawtohex(c.circuit),d.name,s.sid,s.serial#,
c.status, c.queue,c.bytes
from v$circuit c,v$dispatcher d, v$shared_server s1,v$session s
where c.dispatcher = d.paddr(+)
and c.server = s1.paddr(+)
and c.saddr = s.saddr(+)
order by c.circuit;
CIRCUIT
-------C3874F38
C387536C
C38757A0
C3875BD4
C3876008
C387643C
C3876870
C3876CA4
C38770D8
C387750C
C3877940
C3877D74
C38781A8
C38785DC
C3878E44
C38796AC
C3879F14
C387A348
C387A77C
C387ABB0
C387C4E8
C387C91C
C387CD50
C387D184
C387D5B8
C387D9EC
C387E688
C3880C5C
C3881090
C38814C4
NAME SID
SERIAL#
STATUS
----- ---------- ---------- ---------------D002
105
3180 NORMAL
D000
27
9 NORMAL
D000
29
3 NORMAL
D000
NORMAL
D002
84
9198 NORMAL
D002
108
7569 NORMAL
D001
32
15965 NORMAL
D002
93
2617 NORMAL
D002
98
16075 NORMAL
D001
112
28224 NORMAL
D001
31
40294 NORMAL
D001
11
22751 NORMAL
D002
100
17533 NORMAL
D001
87
2306 NORMAL
D002
23
23536 NORMAL
D001
65
21222 NORMAL
D002
102
20336 NORMAL
D001
7
21894 NORMAL
D001
68
40001 NORMAL
D002
18
28880 NORMAL
D001
56
3310 NORMAL
D002
95
11271 NORMAL
D002
69
26258 NORMAL
D002
74
46523 NORMAL
D000
75
12017 NORMAL
D001
44
11978 NORMAL
D002
88
10584 NORMAL
D001
107
3497 NORMAL
D001
41
62817 NORMAL
D001
20
37238 NORMAL
QUEUE
BYTES
---------------- ---------NONE
1601
NONE
95192596
NONE
1720467
NONE
402530056
NONE
1603
NONE
1665
NONE
116787
NONE
2643
NONE
1603
NONE
149265
NONE
4666
NONE
1603
NONE
1603
NONE
2644
NONE
1603
NONE
1601
NONE
611048
NONE
2314
NONE
1665
NONE
2149
NONE
138351
NONE
3475
NONE
1603
NONE
44518043
NONE
8195
NONE
21736
NONE
7911
NONE
1635
NONE
445986
NONE
56800
Dispatcher
Das Diagramm «Dispatcher» zeigt die Statistiken der Dispatcher Prozesse der Datenbank Instance.
SQL-Statement
select name, status, accept, messages, bytes, idle, busy
from v$dispatcher
order by name;
NAME
----D000
D001
D002
Seite
50 von
82
Ausgabe
5. Januar
1995
STATUS
---------------WAIT
WAIT
WAIT
ACC MESSAGES
--- ---------YES
8980779
YES
2080331
YES
4203849
BYTES
---------515802691
75276628
147948411
IDLE
BUSY
---------- ---------114566779
716941
115068281
215441
114892365
391356
Seite -50
$NDGLD $*
Oracle Datenbank Tuning
MTS
Konfiguration und Waiting auf MTS Dispatcher und Server.
COLUMN NAME FORMAT A25 TRUNC
COLUMN NETWORK FORMAT A10 TRUNC
COLUMN VALUE FORMAT A40 TRUNC
REM Actual MTS-Parameters
SELECT NAME, VALUE
FROM V$PARAMETER
WHERE NAME LIKE ’%mts%’ OR NAME LIKE ’%MTS%’;
REM Max. Number of Server-Processes
SELECT * FROM V$MTS
REM Waits of Dispatchers
SELECT NETWORK,DECODE(SUM(TOTALQ),0,’not activ’,
ROUND(SUM(WAIT)/SUM(TOTALQ),2)) AVG Waits (1/100 s)
FROM V$DISPATCHER D, V$QUEUE Q
WHERE Q.TYPE = ’DISPATCHER’
AND D.PADDR = Q.PADDR
GROUP BY NETWORK;
REM Waits of Shared-Server
SELECT DECODE(TOTALQ,0,’not activ’,
ROUND(WAIT/TOTALQ,2))
AVG Waits (1/100 s)
FROM V$QUEUE
WHERE TYPE = ’COMMON’
REM Nbr. Processes actually waiting for a shared server
SELECT QUEUED
FROM V$QUEUE
WHERE TYPE = ’COMMON’
Max.Number of Server-Processes
MAXIMUM_CONNECTIONS SERVERS_STARTED SERVERS_TERMINATED SERVERS_HIGHWATER
------------------- --------------- ------------------ ----------------56
0
0
5
Waits of Dispatchers
NETWORK
---------ipc
tcp
AVG Waits (1/100 s)
---------------------------------------,02
,03
Waits of Shared-Server
AVG Waits (1/100 s)
---------------------------------------,02
Nbr.Processes actually waiting for a shared server
QUEUED
--------0
Wenn SERVERS_HIGHWATER den Wert von mts_max_servers (=20)
erreicht, so muss mts_max_servers erhöht werden.
Ausgabe 5. Januar 1995
Seite 51
von-51
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
File I/O Rate
Details
Das Diagramm «File I/O Rate Details» zeigt die Anzahl der physischen
Reads und Writes pro Sekunde für jedes einzelne Datenbank File.
SQL-Statement
select
from
where
order
substr(NAME,1,30),PHYRDS,PHYWRTS,PHYBLKRD,PHYBLKWRT
V$DBFILE DF, V$FILESTAT FS
DF.FILE#=FS.FILE#
by NAME;
NAME
PHYRDS
PHYWRTS PHYBLKRD PHYBLKWRT
------------------------------ ---------- ---------- ---------- ---------/dev/vg11/rsipp01
1098
545164
1098
545164
/dev/vg11/rsipp02
12288
151119
55335
151119
/dev/vg11/rsipp03
734343
1306301
4463897
1317065
/dev/vg11/rsipp04
1135197
76502 17699546
76502
/dev/vg11/rsipp05
266195
21536
3380626
21536
/dev/vg11/rsipp06
144729
61222
154062
100771
/dev/vg11/rsipp07
250033
787023
357896
909131
/dev/vg11/rsipp08
161868
823112
1122768
1613132
/dev/vg11/rsipp09
0
1499
0
1499
/dev/vg11/rsipp10
0
0
0
0
/sipp/d1/ts_system/P18M7S_sys1
16122
31197
28952
31941
/sipp/d1/ts_system/P18M7S_user
0
0
0
0
Free List Hit%
Das Diagramm «Free List Hit %» zeigt Informationen zur Datenblock
Contention. Für jedes Segment unterhält Oracle ein oder mehrere Freelists. Freelists enthalten allozierte Datenblocks für diesen SegmentExtent mit freiem Platz für INSERTS. Bei vielen concurrent INSERTS sind
unter Umständen mehrere Freelists zu erstellen.
SQL-Statement
select
from
where
and
(sum(value)-(sum(count)/2))/sum(value)*100
v$waitstat w, v$sysstat s
w.class=’free list’
s.name in (’db block gets’, ’consistent gets’);
Berechnung
HitRatio % =
Sum(Count)
Sum(Value)
* 100
select (sum(count) / (sum(value))) * 100
from v$waitstat w, v$sysstat s
where w.class=’free list’
and s.name in (’db block gets’, ’consistent gets’);
Freelist Contention
darf nicht mehr als 1 %
betragen
Seite
52 von
82
Ausgabe
5. Januar
1995
Seite -52
$NDGLD $*
Oracle Datenbank Tuning
Latch
Das Diagramm «Latch» statistics zeigt Low-level Locks an shared internen Memorystrukturen.
SQL-Statement
select substr(ln.name,1,30) "Name",l.gets,l.misses,l.sleeps,
l.immediate_gets "ImmGets",l.immediate_misses "ImmMiss"
from v$latch l, v$latchname ln, v$latchholder lh
where l.latch#=ln.latch#
and l.addr=lh.laddr(+)
order by l.level#, l.latch#;
Name
GETS
MISSES
SLEEPS
ImmGets
ImmMiss
------------------------------ ---------- ---------- ---------- ---------- ---------process allocation
1151
0
0
1151
0
session switching
26746
0
0
0
0
cached attr list
0
0
0
0
0
modify parameter values
100971
17
10
0
0
virtual circuit buffers
61160911
196
181
0
0
session allocation
3502845
19
21
0
0
session idle bit
34993386
451
519
0
0
cache buffers chains
351978424
1138
1338 805523115
2025
global transaction
1926073
0
0
0
0
global tx hash mapping
724195
0
0
1
0
NLS data objects
1
0
0
0
0
global tx free list
260731
0
0
0
0
query server process
0
0
0
0
0
cache buffer handles
3545
0
0
0
0
multiblock read objects
4064844
17
18
9
0
cache buffers lru chain
32308019
1421
2038 29031168
5631
KCL name table latch
0
0
0
0
0
loader state object freelist
2774
0
0
0
0
dml lock allocation
4328384
11
11
0
0
list of block allocation
2346310
3
3
0
0
sort extent pool
13760
0
0
0
0
user lock
19189
0
0
0
0
enqueue hash chains
26905627
598
686
0
0
trace latch
0
0
0
0
0
cache protection latch
0
0
0
0
0
archive control
653
0
0
0
0
lock element parent latch
0
0
0
0
0
row cache objects
7015533
17
29
6391
0
process queue reference
0
0
0
0
0
enqueues
10738911
18
18
0
0
instance latch
0
0
0
0
0
undo global data
5275435
27
28
0
0
library cache
49455756
6235
4896
6375
9
library cache load lock
23658
0
0
0
0
error message lists
0
0
0
0
0
process queue
0
0
0
0
0
device information
0
0
0
0
0
redo copy
0
0
0
0
0
query server freelists
0
0
0
0
0
parallel query alloc buffer
0
0
0
0
0
redo allocation
14726380
865
1039
0
0
shared pool
4605571
46
55
0
0
messages
7238853
615
595
0
0
system commit number
30233743
281
426
0
0
KCL freelist latch
0
0
0
0
0
transaction allocation
4792886
13
18
0
0
sequence cache
62029
0
0
0
0
cost function
1
0
0
0
0
virtual circuit queues
58346794
2263
2111
0
0
virtual circuits
15239348
69
76
0
0
parallel query stats
0
0
0
0
0
latch wait list
9097
0
0
0
0
ktm global data
4584
0
0
0
0
Ausgabe 5. Januar 1995
Seite 53
von-53
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
Library Cache
Details
Das Diagramm «Library Cache Details» zeigt Detailinformationen des
Library Cache im Shared Pool der Instance. Der Library Cache enthält
SQL und PL/SQL Code in geparster Form. Es ist wichtig, dass die Ratio
für diese Bereiche nahezu 100% beträgt.
SQL-Statement
select namespace,gets,gethits,
round(gethitratio*100) "RATIO%",
pins,pinhits,round(pinhitratio*100) "RATIO%"
from v$librarycache
order by namespace;
NAMESPACE
GETS
GETHITS
RATIO%
PINS
PINHITS
RATIO%
--------------- ---------- ---------- ---------- ---------- ---------- ---------BODY
7216
7172
99
7216
7166
99
CLUSTER
171
84
49
159
77
48
INDEX
852
540
63
534
380
71
OBJECT
0
0
100
0
0
100
PIPE
21167
21166
100
22158
22157
100
SQL AREA
1910800
1885069
99 11932152 11870605
99
TABLE/PROCEDURE
228485
220990
97
458154
446343
97
TRIGGER
2736
2733
100
2736
2729
100
Lock
Das Diagramm «Lock» zeigt aktive Locks der Datenbank Instance. Eine
bessere Übersicht der Locksituation bietet der Lockmanager an, der auch
wartende Sessions in der Treelist anzeigt.
SQL-Statement
select s.username, s.sid, s.serial#, l.type, l.id1, l.id2,
decode(l.lmode, 1, ’NULL’, 2, ’ROW SHARE’, 3,
’ROW EXCLUSIVE’, 4, ’SHARE’, 5, ’SHARE ROW EXCLUSIVE’, 6,
’EXCLUSIVE’, ’?’), decode(l.request, 1, ’NULL’, 2, ’ROW SHARE’, 3,
’ROW EXCLUSIVE’, 4,’SHARE’, 5, ’SHARE ROW EXCLUSIVE’, 6,
’EXCLUSIVE’, ’?’)
from v$lock l, v$session s
where l.sid=s.sid
order by s.sid, l.type;
Verursacher des
Locks
Wartende Session
Seite
54 von
82
Ausgabe
5. Januar
1995
Seite -54
$NDGLD $*
Oracle Datenbank Tuning
Wartende Sessions
Die View ROW_LOCK_WAITS zeigt die wartenden Sessions. Dieses
Statement ist als View implementiert, es darf keine Rows zurückbringen,
da sonst ein User warten muss.
create or replace view row_lock_waits
(username, sid, object_owner,
object_name, sql_text, file_nr, block_nr, record_nr)
as
select s.username, s.sid,o.owner,o.object_name,a.sql_text,
s.row_wait_file#,s.row_wait_block#,s.row_wait_row#
from v$session s, v$sqlarea a, dba_objects o
where o.object_id = s.row_wait_obj#
and s.sql_address = a.address
and s.row_wait_obj# > 0;
select * from row_lock_waits;
Memory
Allocated
Das Diagramm «Memory Allocated» zeigt das allozierte Memory in Bytes
für die gesamte Instance (alle Sessions).
SQL-Statement
select
from
where
and
/*+ use_nl(n,s) */ sum(value)
v$statname n, v$sesstat s
s.statistic# = n.statistic#
name = ’session uga memory’;
Zur Zeit sind 990952
Bytes alloziert
Die Gösse der gesamten SGA kann mittels Show SGA im Server Manager angezeigt werden:
SVRMGR> show sga
Total System Global Area
Fixed Size
Variable Size
Database Buffers
Redo Buffers
Ausgabe 5. Januar 1995
114220552
38904
50243088
62914560
1024000
Byte
Byte
Byte
Byte
Byte
Seite 55
von-55
82
Seite
Oracle Datenbank Tuning
Zur Verfügung
stehendes Memory
$NDGLD $*
Wenn mehr Memory für die SGA alloziert wird, so muss das aktuell vorhandene, freie Memory verifiziert werden. Dazu eignet sich das
Programm top.
freies
Memory
Seite
56 von
82
Ausgabe
5. Januar
1995
Seite -56
$NDGLD $*
Oracle Datenbank Tuning
Network
Bytes Rate
Das Diagramm «Network Bytes Rate» zeigt die Anzahl Bytes / Sekunde
an Daten, die vom Datenbank Server und seinen Clients über SQL*Net
ausgetauscht werden.
SQL-Statement
select sum(value) from v$sysstat
where name like ’bytes%SQL*Net%’;
Network I/O Rate
Das Diagramm «Network I/O Rate» zeigt die Anzahl Message Packete /
Sekunde die vom Datenbank Server und seinen Clients über SQL*Net
ausgetauscht werden.
SQL-Statement
select sum(total_waits)
from v$system_event
where event like ’SQL*Net%’;
Ausgabe 5. Januar 1995
Seite 57
von-57
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
Session Events
Die V$SESSION_WAIT View enthält sämtliche Events, welche die Userund System Sessions in den Wartezustand versetzen. Diese View kann
verwendet werden, um rasch einen Performance Engpass
herauszufinden.
SQL-Statement
select w.Sid "Sid", nvl(substr(s.username,1,15),’Background’) "User",
substr(w.event,1,25) "Event", w.wait_time "Wait Time"
from v$session_wait w, v$session s
where w.sid = s.sid
order by 2,4;
Sid
User
---------- --------------1 Background
2 Background
3 Background
19 Background
13 Background
54 Background
70 Background
5 Background
111 Background
6 Background
4 Background
92 CLIENT
41 CLIENT
84 CLIENT
110 CLIENT
32 CLIENT
30 CLIENT
28 CLIENT
29 DBSNMP
27 DBSNMP
7 EASY2REG
31 EASY2REG
87 EASY2REG
98 EASY2REG
88 EASY2REG
57 EASY2REG
44 EASY2REG
56 EASY2REG
23 EASY2REG
11 EASY2REG
59 EASY2REG
95 EASY2REG
76 EASY2REG
69 EASY2REG
114 EASY2REG
100 EASY2REG
78 OPS$SIPPALT
64 OPS$SIPPROC
24 OPS$SIPPROC
12 OPS$SIPPROC
21 OPS$SIPPROC
80 SIPP
43 SIPP
103 SIPP
106 SIPP
75 SIPP
18 SIPP
112 SYSADM
Event
Wait Time
------------------------- ---------pmon timer
0
rdbms ipc message
0
rdbms ipc message
0
rdbms ipc message
0
rdbms ipc message
0
rdbms ipc message
0
rdbms ipc message
0
smon timer
0
rdbms ipc message
0
rdbms ipc message
0
log file parallel write
0
db file scattered read
0
SQL*Net message from clie
0
SQL*Net message from clie
0
SQL*Net message from clie
0
SQL*Net message from clie
0
SQL*Net message from clie
0
SQL*Net message from clie
0
SQL*Net message from clie
0
SQL*Net message from clie
0
SQL*Net message from clie
0
SQL*Net message from clie
0
SQL*Net message from clie
0
SQL*Net message from clie
0
SQL*Net message from clie
0
SQL*Net message from clie
0
SQL*Net message from clie
0
SQL*Net message from clie
0
SQL*Net message from clie
0
SQL*Net message from clie
0
SQL*Net message from clie
0
SQL*Net message from clie
0
SQL*Net message from clie
0
SQL*Net message from clie
0
SQL*Net message from clie
0
SQL*Net message from clie
0
db file scattered read
-1
log file sync
0
SQL*Net message from clie
0
pipe get
0
SQL*Net message from clie
0
SQL*Net message from clie
0
SQL*Net message from clie
0
SQL*Net more data from db
0
SQL*Net message from clie
0
SQL*Net message from clie
0
SQL*Net message from clie
18
SQL*Net message from clie
0
Eine Waiting Time von 0 zeigt an, dass die Session gerade auf einen
Event wartet. Grosse Wait Times weisen auf ein Performance Problem
hin (siehe Oracle Tuning Guide Seite A-4).
Seite
58 von
82
Ausgabe
5. Januar
1995
Seite -58
$NDGLD $*
Oracle Datenbank Tuning
Parse Ratio
Das Diagramm «Parse Ratio» zeigt, wie gross der Parsing Aufwand der
SQL-Statements ist.
SQL-Statement
select sum(decode(name,’parse count’, value,0)) /
sum(decode(name,’opened cursors cumulative’, value,0))
from v$sysstat;
Processes
Das Diagramm «Process» zeigt Informationen zu den Oracle Background
und User Prozessen.
SQL-Statement
select substr(p.pid,1,4) "Pid", p.spid "Spid",
substr(p.username,1,15) "User",
decode(s.terminal, NULL, p.terminal, s.terminal) "Terminal",
decode(s.program, NULL, p.program, s.program) "Program"
from v$process p, v$session s
where p.addr=s.paddr order by pid;
Queue
Das Diagramm «Queue» zeigt die Aktivitäten des Multi-Threaded
Servers.
SQL-Statement
select rawtohex(paddr), type, queued, totalq,
decode(totalq, 0, 0, wait/totalq/100)
from v$queue order by paddr;
OUTBOUND = Used by Remote Servers
COMMON = Processed by Servers
Ausgabe 5. Januar 1995
Seite 59
von-59
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
Redo Alloc Hit%
Das Diagramm «Redo Allocation Hit %» zeigt das Buffer Tuning der
Redolog File Aktivitäten. Die Misses dürfen nicht grösser als 1 % sein.
Berechnung
HitRatio % =
Gets + ImmediateGets
Gets + ImmediateGets + Misses + ImmediateMisses
* 100
SQL-Statement
select ((gets+immediate_gets) /
(gets+immediate_gets+misses+immediate_misses)) *100
from v$latch
where name = ’redo allocation’;
Redo Statistics
Das Diagramm «Redo Statistics» zeigt die Anzahl Redo Entries, Space
Requests und Synch. Writes pro Sekunde für die Datenbank Instance.
SQL-Statement
select sum(decode(name,’redo entries’, value,0)),
sum(decode(name,’redo log space requests’, value,0)),
sum(decode(name,’redo synch writes’, value,0))
from v$sysstat;
Seite
60 von
82
Ausgabe
5. Januar
1995
Seite -60
$NDGLD $*
Oracle Datenbank Tuning
Session
Das Diagramm «Session» zeigt die vorhandenen User Sessions mit ihren
SID’s, Serial-Nr und PID’s.
SQL-Statement
select s.sid, s.serial#,p.pid,s.status,s.username,lockwait,
decode(s.command,0,’NONE’,NVL(a.name,’UNKNOWN’))
from v$session s, v$process p, audit_actions a
where s.paddr=p.addr
and a.action(+)=s.command
order by s.sid, s.serial#;
Sort Rows Rate
Das Diagramm «Sort Rows Rate» zeigt die Sort Statistiken für die
gesamte Instance.
SQL-Statement
select value from v$sysstat
where name = ’sorts (rows)’;
Sqlarea
Das Diagramm «SQL Area» zeigt die shared Cusor Informationen im
Library Cache.
SQL-Statement
select
from
where
and
order
distinct nvl(username,type) username,sid,sql_text
v$session, v$sqlarea
sql_hash_value = hash_value
sql_text is not null
by username;
SQL of connected sessions
USERNAME
SID SQL_TEXT
------------ ---- ----------------------------------------------------------CLIENT
30 select * from OWNERSHIP, IMEI, EQUIPTYPE where
OWNERSHIP.imei_id = IMEI.imei_id and IMEI.equip_id =
EQUIPTYPE.equip_id and cus_id = ’0’ order by manufacturer,
description
CLIENT
59 select * from USERS where username = ’kon0’
System
Statistics
Das Diagramm «System Stats» zeigt alle Parameter der wichtigen
System Statistiktabelle v$sysstat.
SQL-Statement
select s.name, s.value
from v$sysstat s
order by s.name, s.value;
NAME
---------------------------------------------------------------background checkpoints completed
background checkpoints started
background timeouts
bytes received via SQL*Net from client
bytes received via SQL*Net from dblink
bytes sent via SQL*Net to client
bytes sent via SQL*Net to dblink
calls to get snapshot scn: kcmgss
calls to kcmgas
calls to kcmgcs
calls to kcmgrs
change write time
cleanouts and rollbacks - consistent read gets
cleanouts only - consistent read gets
cluster key scan block gets
cluster key scans
commit cleanout failure: write disabled
commit cleanout failures: buffer being written
commit cleanout failures: callback failure
..............
Ausgabe 5. Januar 1995
VALUE
---------888
888
1001617
883419229
1304228985
4092682369
929074178
13330319
1382814
89286
20090087
145527
61807
24494
875660
408030
0
1217
38
Seite 61
von-61
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
Auswertung der
System Statistik
Aus den Systemstatistiken können wichtige Informationen gewonnen
werden. Man beachte dass es sich bei diesen Angaben immer um kumulierte Werte seit dem letzten Startup handelt.
Full Table Scans
table
table
table
table
scan blocks gotten
scan rows gotten
scans (long tables)
scans (short tables)
61’900’307
194’6840’695
13’267
307’195
Index Scans
table fetch by rowid
15’653’655
Redo Waits
redo log space requests
redo log space wait time
1018
21263
Bei grösseren Waits sind die RDO-Files zu vergrössern und er Parameter
LOG_BUFFER muss erhöht werden. Die Zeit ist in 1/100 Sekunden
angegeben (21263 = 212 Sekunden = 3,5 Minuten in etwa 5 Wochen).
Seite
62 von
82
Ausgabe
5. Januar
1995
Seite -62
$NDGLD $*
Oracle Datenbank Tuning
Table Access
Das Diagramm «Table Access» zeigt alle Datenbankobjekte, welche zur
Zeit von welcher Session benutzt werden.
SQL-Statement
select sid "Sid", substr(owner,1,15) "Owner",
substr(object,1,20) "Object"
from v$access
where owner != ’SYS’
order by owner;
Sid
---------31
49
57
92
92
84
78
78
78
78
Owner
--------------CLIENT
CLIENT
CLIENT
CLIENT
CLIENT
CLIENT
OPS$SIPPALT
OPS$SIPPALT
OPS$SIPPALT
OPS$SIPPALT
Object
-------------------USERS
USERS
USERS
USERS
MSISDN
USERS
ACM
SEQ_ORDER_REQ_ORD_ID
MSISDN
MESSAGE
Users logged on
Das Diagramm «No. of Users Logged On» zeigt die Anzahl concurrent
Users Sessions, unabhänig davon ob sie nun aktiv sind oder nicht.
SQL-Statement
select sessions_current
from v$license;
Users running
Das Diagramm «No. of Users Running» zeigt die Users Sessions, welche
eine Transaktion ausführen.
SQL-Statement
select count(*)
from v$session_wait
where wait_time!=0;
Users waiting
Das Diagramm «No. of Users Waiting» zeigt die User Sessions, die auf
einen Event (for whatever reason) warten müssen, um eine Aktion
durchzuführen.
SQL-Statement
select substr(w.sid,1,5) "Sid",
substr(s.username,1,15) "User",
substr(event,1,30) "Event",
seconds_in_wait "Wait [s]"
from v$session_wait w, v$session s
where s.sid = w.sid
and state = ’WAITING’
and event not like ’SQL*Net%’
and event != ’client message’
and event not like ’%mon timer’
and event != ’rdbms ipc message’
and event != ’Null Event’;
Sid User
Event
Wait [s]
----- --------------- ------------------------------ ---------12 OPS$SIPPROC
pipe get
48
Users waiting
for Locks
Das Diagramm «No. of Users Waiting For Lock» zeigt die User Sessions,
die auf die Freigabe eines Locks warten müssen.
SQL-Statement
select count(*)
from v$session
where lockwait is not null;
Ausgabe 5. Januar 1995
Seite 63
von-63
82
Seite
Oracle Datenbank Tuning
Object Status
Seite
64 von
82
Ausgabe
5. Januar
$NDGLD $*
Um sicher zu sein, dass alle Objekte den Status = VALID haben, kann
das folgende Query benutzt werden.
1995
Seite -64
$NDGLD $*
Oracle Datenbank Tuning
4.2
Detaillierte Analyse des Shared Pools
Das dringendste Problem im Zusammenhang mit dem Shared Pool bildet
die Fragmentierung des Memory in kleine «Stücke» im Laufe des
Betriebs. Oracle hat verschiedene Massnahmen getroffen, um diese
Fragmentierung zu reduzieren:
• Memory wird nicht mehr in grossen Chunks alloziert.
• Reduzieren des Memory Bedarfs für das Per-User Memory (UGA).
• Keeping von grossen PL/SQL Teilen im Memory, um das Flushing von
anderen Objekten aus dem Memory zu verhindern.
Memory
Fragmentierung
Jeder Versuch ein grosses PL/SQL Package zu laden, hat zur Folge dass
andere Objekte aus dem Pool geflushed werden, was zu einem
ORA-4031 Error führen kann.
Shared Pool
SHARED_POOL_SIZE
SHARED_POOL_RESERVED_SIZE
Flushing
< SHARED_POOL_RESERVED_MIN_ALLOC
> SHARED_POOL_RESERVED_MIN_ALLOC
Objekte die kleiner sind als SHARED_POOL_RESERVES_MIN_ALLOC,
kommen in den «linken» Teil, Objekte die grösser sind als diesen Parameter kommen in den rechten Teil des Shared Pools.
Erkennen des
Flushings
In der Fixed Tabelle X$KSMLRU wird vermerkt, welche Objekte ein Flush
anderer Objekte auslösen. Diese Tabelle kann dazu verwendet werden,
die Objekte zu identifizieren, welche für die Fragmentierung
verantwortlich sind.
COLUMN
COLUMN
COLUMN
COLUMN
ksmlrnum
ksmlrcom
ksmlrhon
ksmlrsiz
HEADING
HEADING
HEADING
HEADING
"Flushed|SP-Pieces" FORMAT 999,990
"Comment"
"Command"
"Size"
SELECT ksmlrhon,ksmlrcom,ksmlrsiz,ksmlrnum
FROM sys.x$ksmlru;
Ausgabe 5. Januar 1995
Seite 65
von-65
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
Auswertung
ksmlrnum
ksmlrcom
ksmlrhon
ksmlrsiz
Anzahl Objekte, die aus dem Shared Pool geflushed wurden infolge Memory Allocation des geladenen Objekts.
(Flushed SP-Pieces).
Allocation Kommentar, welches die Art der Memory Allocation umschreibt. Angaben wie ’MPCODE’ oder
’PLSQL%’ heissen, dass ein grosses PL/SQL Objekt geladen wurde (Comment).
Name des Objects, das geladen wurde, nur bei Pl/SQL
und Cursor Objekten (Command).
Grösse des Contiguous Memorybereichs der alloziert
wurde (Size).
Werte > 5K
Start to be a Problem.
Werte > 10K
Serious Problem
Werte > 20K
Very serious Problem
Flushed
Command
Comment
Size SP-Pieces
----------------------------- -------------------- ---------- --------object info
8664
506
KXHF Slot Table
19528
513
0
0
0
0
0
0
0
0
0
0
0
0
0
0
0
0
Im obigen Fall haben wir es also bereits mit einem very serious Problem zu tun !
Beachte !
Seite
66 von
82
Ausgabe
5. Januar
Diese Tabelle darf nicht mehrmals kurz hintereinander abgefragt werden,
da sie immer nur die letzten Angaben enthält. Nach dem Abfragen sind
keine Angaben mehr in der Tabelle vorhanden !
1995
Seite -66
$NDGLD $*
Oracle Datenbank Tuning
Vorhandene Memory
Segmente
Die Tabelle X$KSMSP gibt Auskunft über die Memory Segmente und
deren Stati im Shared Pool.
Sum:
State:
Max:
SQL-Statement
Total sum(size) of all memory segments of this type
recr
= Flushable segment.
freeabl = This segment is in use and can't be flushed.
free
= Free segment
perm = Permanently allocated memory segment
This shows the biggest memory segments of this type.
SQL-Statement zur Auslistung der Memory Segmente
SELECT SUBSTR(ksmchcom,1,16) ksmchcom,
DECODE(ksmchcls, ’recr’,
’Flushable’,
’freeabl’, ’InUse’,
’free’,
’Free’,
’perm’,
’Permanent’,
ksmchcls) text_ksmsp,
SUM(ksmchsiz) sum_ksmsp,
MAX(ksmchsiz) max_ksmsp,
COUNT(*) count_ksmsp
FROM sys.x$ksmsp
GROUP BY ksmchcom,ksmchcls
ORDER BY ksmchcom,ksmchcls;
****** Memory Segments and their state in the SHARED POOL *******************
Description
State
Sum (Size)
Max (Size)
Number of
[Bytes]
[Bytes]
Segments
---------------- ------------- ----------------- --------------- -------------character set m InUse
32896
15152
9
dictionary cach InUse
1376624
4256
814
fixed allocatio Flushable
240
40
6
free memory
Free
6258376
6216568
743
free memory
R-free
8000000
8000000
1
KGL handles
Flushable
1596184
960
5272
kxfp subheap
Flushable
584
584
1
kzull
InUse
304
64
6
library cache
InUse
2545592
584
8643
library cache
Flushable
2210656
584
4708
multiblock rea InUse
560
280
2
permanent memor Permanent
16420872
16420872
1
PLS cca hp desc InUse
168
168
1
PLS non-lib hp Flushable
2104
2104
1
PL/SQL DIANA
InUse
1044264
2112
626
PL/SQL DIANA
Flushable
114392
2104
72
PL/SQL MPCODE
InUse
449968
8400
179
PL/SQL MPCODE
Flushable
97344
2104
90
reserved stoppe R-freea
48
24
2
row cache lru
Flushable
1944
64
40
session param v InUse
31440
2096
15
sql area
InUse
7230464
4256
2333
sql area
Flushable
2559104
4264
808
table definiti Flushable
64
64
1
trigger defini Flushable
1080
1080
1
library cache
InUse
2545592
584
8643
library cache
Flushable
2210656
584
4708
multiblock rea InUse
560
280
2
permanent memor Permanent
16420872
16420872
1
PLS cca hp desc InUse
168
168
1
PLS non-lib hp Flushable
2104
2104
1
PL/SQL DIANA
InUse
1044264
2112
626
PL/SQL DIANA
Flushable
114392
2104
72
PL/SQL MPCODE
InUse
449968
8400
179
PL/SQL MPCODE
Flushable
97344
2104
90
reserved stoppe R-freea
48
24
2
Ausgabe 5. Januar 1995
Seite 67
von-67
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
Grosse Objecte
lokalisieren
Grosse Objecte in Shared Pool führen zu Problemen. Sie haben zur
Folge, dass andere kleinere Objekte geflushed werden. Damit grosse
Objekte nicht mehr jedesmal geladen werden müssen und so kleinere
Objekte verdrängen was zu einer Fragmentierung führt, können solche
Objekte als ’keep’ markiert werden. Sie werden dann nach dem erstmaligen Laden nicht mehr geflushed. Ab Version 7.3 müssen grosse Objekte
nicht mehr als ’keep’ markiert werden, da allgemein grosse Objekte in
kleine Einheiten zerlegt werden.
Kontrolle, welche grossen Objekte im Shared Pool sind:
SELECT owner, name, type, kept, loads, sharable_mem
FROM v$db_object_cache
WHERE sharable_mem > 10000
ORDER BY sharable_mem desc;
Auswertung
Alle Objekte im Shared Pool welche grösser als 10 KBytes sind:
Owner
-------SYS
SYS
SYS
SIPP
SYS
SYS
SYS
SYS
SYS
SYS
SYS
SYS
SYS
SYS
SYS
SYS
SYS
SYS
SYS
Grosse Anonymous
PL/SQL Blocks
Name
-----------------------------STANDARD
STANDARD
INDREORG
CHECK_ACCM_MAX_TOTAL
DBMS_SYS_SQL
DBMS_SHARED_POOL
V$SQLAREA
DBMS_UTILITY
DBMS_IJOB
V$DB_OBJECT_CACHE
DBMS_PIPE
DBMS_STANDARD
DBMS_IJOB
DBMS_OUTPUT
DBMS_UTILITY
DBMS_EXPORT_EXTENSION
DBMS_APPLICATION_INFO
DBMS_SQL
DBMS_DEFER_IMPORT_INTERNAL
Type
------------PACKAGE
PACKAGE BODY
PROCEDURE
PROCEDURE
PACKAGE BODY
PACKAGE
VIEW
PACKAGE
PACKAGE
VIEW
PACKAGE
PACKAGE
PACKAGE BODY
PACKAGE
PACKAGE BODY
PACKAGE
PACKAGE
PACKAGE BODY
PACKAGE
Size
Kept Loads
Bytes
---- ----- -----------NO
1
120,628
NO
1
28,700
NO
1
25,188
NO
1
23,505
NO
1
22,448
NO
1
22,244
NO
1
19,597
NO
1
19,592
NO
1
16,853
NO
1
16,629
NO
1
15,853
NO
1
14,521
NO
1
13,853
NO
1
13,703
NO
1
13,480
NO
1
12,561
NO
1
12,129
NO
1
11,900
NO
1
10,798
Grosse anonymous PL/SQL Blocks sollten in kleine Packages unterteilt
werden, welche Funktionen aufrufen.
Identifikation von grossen anonymous PL/SQL Blocks:
COLUMN sql_text FORMAT A60 HEADING ’SQL-Text > 500 Bytes’
SELECT sql_text FROM v$sqlarea
WHERE command_type = 47
-- Command type for anonymous block
AND length(sql_text) > 500;
Seite
68 von
82
Ausgabe
5. Januar
1995
Seite -68
$NDGLD $*
Oracle Datenbank Tuning
Grösse des Shared
Pools bestimmen
Die Grösse des Shared Pools ist abhängig von der Applikation und kann
meist erst während der Produktion festgelegt werden.
SHARED_POOL_SIZE = ?
DB-Objects
Memory im Shared Pool für die Datenbank Objekte wie Packages.
COLUMN total_mem FORMAT 999,999,999 HEADING ’Space needed for DB-Objects in SP’
SELECT SUM(sharable_mem) total_mem FROM v$db_object_cache;
Space needed for DB-Objects in SP
--------------------------------836,096
SQL
Memory im Shared Pool für normale SQL-Statements.
COLUMN total_mem FORMAT 999,999,999 HEADING ’Space needed for SQL in SP’
SELECT SUM(sharable_mem) total_mem FROM v$sqlarea;
Space needed for SQL in SP
-------------------------25,617,655
Per-User and
Per-Cursor
Pro concurrent User wird ca 250 Bytes benötigt.
COLUMN total_mem FORMAT 999,999,999 HEADING ’Space needed for Users in SP’
SELECT SUM(250 * users_opening) total_mem FROM v$sqlarea;
Space needed for Users in SP
---------------------------15,125
MTS
Es muss genügend Memory für jeden MTS-User (Multithreaded Server)
reserviert werden, der sich über SQL*NET connected.
COLUMN
SELECT
FROM
WHERE
AND
total_mem FORMAT 999,999,999 HEADING ’Space needed for MTS in SP’
SUM(value) total_mem
v$sesstat s, v$statname n
s.statistic# = n.statistic#
n.name = ’session uga memory max’;
Space needed for MTS in SP
-------------------------7,234,992
Dies ergibt Total DB-Objects + SQL + User + MTS
836,096 + 25,617,655 + 15,125 + 7,234,992 = 33,703,868
Zuzüglich 30 % Overhead
= 10,111,160
------------------------------------------------------Total
40,815,028
=======================================================
Ausgabe 5. Januar 1995
Seite 69
von-69
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
Grösse des Reserved
Pools bestimmen
Die Grösse des Reserved Pools ist abhängig von der Grösse und Menge
der zu erwartenden grossen Memory Segmente, welche in der Applikation auftreten werden. Der optimale Wert kann meist erst während der
Produktion festgelegt werden.
SHARED_POOL_RESERVED_SIZE=?
(ReservedList)
Zur Auswertung kann die View V$SHARED_POOL_RESERVED beigezogen werden.
Folgende Attribute der View enthalten Werte nur wenn der INIT.ORA
Parameter SHARED_POOL_RESERVED_SIZE gesetzt ist.
SQL> desc v$shared_pool_reserved;
Column
Description
FREE_SPACE
AVG_FREE_SIZE
FREE_COUNT
MAX_FREE_SIZE
USED_SPACE
AVG_USED_SIZE
USED_COUNT
MAX_USED_SIZE
REQUESTS
is the total amount of free space on the reserved list.
is the average size of the free memory on the reserved list.
is the number of free pieces of memory on the reserved list.
is the size of the largest free piece of memory on the reserved list.
is the total amount of used memory on the reserved list.
is the average size of the of the used memory on the reserved list.
is the number of used pieces of memory on the reserved list.
is the size of the largest used piece of memory on the reserved list.
is the number of times that the reserved list was searched for a
free piece of memory.
REQUEST_MISSES is the number of times the reserved list didn’t have a free piece of
memory to satisfy the request, and proceeded to start flushing
objects from the LRU list.
LAST_MISS_SIZE
is the request size of the last REQUEST_MISS.
MAX_MISS_SIZE
is the request size of the largest REQUEST_MISS.
Folgende Attribute haben immer einen definierten Wert, auch wenn der
INIT.ORA Parameter SHARED_POOL_RESERVED_SIZE nicht gesetzt ist.
Column
Description
REQUEST_FAILURES
is the number of times that no memory was found to
satisfy a request (e.g., number of times ORA-4031
occurred).
LAST_FAILURE_SIZE
is the request size of the last failed request (e.g., the
request size of last ORA-4031).
ABORTED_REQUEST_THRESHOLD is the minimum size of a request which will signal an
ORA-4031 error without flushing objects. See the
procedure aborted_request_threshold described above.
LAST_ABORTED_SIZE
is the last size of the request which returned an
ORA-4031 error without flushing objects from the LRU list.
Seite
70 von
82
Ausgabe
5. Januar
1995
Seite -70
$NDGLD $*
Oracle Datenbank Tuning
Tuning Tips
Folgende Tuning Tips können angewendet werden:
• Initiale Grösse von SHARED_POOL_RESERVED_SIZE:
Dieser Wert kann zu Beginn des Tunings auf 10 % des Werts von
SHARED_POOL_SIZE gesetzt werden.
• Initiale Grösse von SHARED_POOL_RESERVED_MIN_ALLOC
Dieser Wert kann zu Beginn des Tunings auf den Default Wert (5K)
gesetzt werden. Wird dieser Wert erhöht, so wird weniger Memory im
Bereich des reservierten Teils des Memory (= Reserved List) alloziert,
dafür mehr im normalen Shared Pool.
• Tuning von SHARED_POOL_RESERVED_SIZE:
Die Reserved List sollte so gross gemacht werden, damit keine Objecte
aus dem Pool geflushed werden müssen. Im Idealfall muss
REQUEST_MISS = 0 angestrebt werden. Auf keine Fall darf
REQUEST_MISS im Laufe der Zeit kontinuierlich zunehmen, ansonsten wird ein ORA-4031 Fehler eintreten.
• Ist SHARED_POOL_RESERVED_SIZE zu klein ?
Sobald der Wert von REQUEST_FAILURES > 0 ist, ist die Reserved
List zu klein und muss vergrößert werden.
• Ist SHARED_POOL_RESERVED_SIZE zu gross ?
Es macht auch keinen Sinn die Reserved List zu gross zu machen. Dies
erkennt man wenn REQUEST_MISS = 0 und FREE_MEMORY > 50%
von SHARED_POOL_RESERVED_SIZE ist.
• Ist SHARED_POOL_SIZE zu klein ?
Dies ist der Fall wenn REQUEST_FAILURES > 0 und
LAST_FAILURE_SIZE < SHARED_POOL_RESERVED_MIN_ALLOC
ist. In diesem Fall muss der Shared Pool vergrössert werden.
Ausgabe 5. Januar 1995
Seite 71
von-71
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
Überwachung des
Shared Pool
Die in diesem Kapitel vorgestellten Regeln verpackt man am besten in
eine Stored Procedure, welche einmal pro Tag gestartet wird. Damit können ORA-4031 Fehler zwar nicht verhindert werden, eine Früherkennung
der Fragmentierung ist damit aber möglich.
Stored Procedure check_shared_pool()
CREATE OR REPLACE
Procedure CHECK_SHARED_POOL IS
/* Local variables and declarations */
l_fhd
l_fdir
l_sps
l_sprs
l_sprma
l_fname
l_fmode
l_open_time
l_current_time
l_global_name
l_sum_flush
l_sum_inuse
l_sum_free
l_sum_perm
l_sum_count
l_size
l_sum_mts
l_sum_obj
l_sum_sql
l_sum_users
l_count
l_message
l_updt_status
utl_File.File_Type;
v$parameter.value%Type;
v$parameter.value%Type;
v$parameter.value%Type;
v$parameter.value%Type;
VARCHAR2(30);
VARCHAR2(1) := ’W’;
VARCHAR2(30);
VARCHAR2(30);
VARCHAR2(30);
NUMBER(15) := 0;
NUMBER(15) := 0;
NUMBER(15) := 0;
NUMBER(15) := 0;
NUMBER(15) := 0;
NUMBER(15) := 0;
NUMBER(15) := 0;
NUMBER(15) := 0;
NUMBER(15) := 0;
NUMBER(15) := 0;
INTEGER := 0;
VARCHAR2(100);
NUMBER(1) := 0;
TYPE opstsp_TR IS TABLE OF x$kghlu%ROWTYPE
INDEX BY BINARY_INTEGER;
l_kghlu
opstsp_TR;
/* Show memory segments and their state in the shared pool */
CURSOR ksmsp_cur IS
SELECT SUBSTR(ksmchcom,1,16) ksmchcom,
DECODE(ksmchcls, ’recr’,
’Flushable’,
’freeabl’, ’InUse’,
’free’,
’Free’,
’perm’,
’Permanent’,
ksmchcls) text_ksmsp,
SUM(ksmchsiz) sum_ksmsp,
MAX(ksmchsiz) max_ksmsp,
COUNT(*) count_ksmsp
FROM sys.x$ksmsp
GROUP BY ksmchcom,ksmchcls
ORDER BY ksmchcom,ksmchcls;
/* Large objects in shared pool (size > 10 KBytes) */
CURSOR big_objects_cur IS
SELECT owner, name, type, kept, loads, sharable_mem
FROM v$db_object_cache
WHERE sharable_mem > 10000
ORDER BY sharable_mem desc;
/* Track allocations in the shared pool that cause */
/* other objects in the shared pool to flushed out */
Seite
72 von
82
Ausgabe
5. Januar
1995
Seite -72
Oracle Datenbank Tuning
$NDGLD $*
CURSOR flushed_obj_cur IS
SELECT NVL(ksmlrhon,’Unknown’) ksmlrhon,
NVL(ksmlrcom,’Not defined’) ksmlrcom,
ksmlrsiz,ksmlrnum
FROM sys.x$ksmlru;
/* Tuning of SHARED_POOL_RESERVED_SIZE */
CURSOR spr_cur IS
SELECT * FROM v$shared_pool_reserved;
BEGIN
/* Open logfile */
BEGIN
SELECT value
INTO l_fdir
FROM v$parameter
WHERE name = ’utl_file_dir’;
IF (l_fdir IS NULL) THEN
RAISE NO_DATA_FOUND;
END IF;
l_fname := ’chksp_’ || TO_CHAR(SYSDATE,’DDMMYYYYHH24MI’);
l_fhd := utl_File.FOpen(l_fdir,l_fname,l_fmode);
EXCEPTION
WHEN NO_DATA_FOUND THEN
RAISE NO_DATA_FOUND;
END;
/* Report Header */
SELECT TO_CHAR(TO_DATE(open_time,’MM/DD/YY HH24:MI:SS’),’DD.MM.YYYY HH24:MI’),
TO_CHAR(SYSDATE,’DD.MM.YYYY HH24:MI’)
INTO l_open_time, l_current_time
FROM v$thread;
SELECT global_name
INTO l_global_name
FROM global_name;
utl_File.Put_Line(l_fhd,’------------------------- Shared Pool statistics -----------------------------’);
utl_File.Putf(l_fhd,’
%s - %s\n’,l_open_time,l_current_time);
utl_File.Putf(l_fhd,’
%s\n’,l_global_name);
utl_File.Put_Line(l_fhd,’-------------------------------------------------------------------------------’);
utl_File.Put_Line(l_fhd,’’);
/* Current size of the SHARED POOL */
SELECT value INTO l_sps
FROM v$parameter
WHERE name = ’shared_pool_size’;
SELECT value INTO l_sprs
FROM v$parameter
WHERE name = ’shared_pool_reserved_size’;
SELECT value INTO l_sprma
FROM v$parameter
WHERE name = ’shared_pool_reserved_min_alloc’;
utl_File.Put_Line(l_fhd,’*********************** Current Size of SHARED POOL ***************************’);
utl_File.Put_Line(l_fhd,’ ’);
utl_File.Putf(l_fhd,’Current size of SHARED_POOL_SIZE ............. = %s [Bytes]\n’, LPAD(l_sps,10,’ ’));
utl_File.Putf(l_fhd,’Current size of SHARED_POOL_RESERVED_SIZE .... = %s [Bytes]\n’, LPAD(l_sprs,10,’ ’));
utl_File.Putf(l_fhd,’Current size of SHARED_POOL_RESERVED_MIN_ALLOC = %s [Bytes]\n’, LPAD(l_sprma,10,’ ’));
Ausgabe 5. Januar 1995
Seite 73
von-73
82
Seite
Oracle Datenbank Tuning
$NDGLD $*
/* -------------------------------------------------------------------------How much space is used in the SHARED POOL ?
One very difficult judgement that needs to be make in oracle7 is to determine
the proper size of the shared pool. The following provides some guidelines
for this. It should be emphasized that these are just guidelines, there are
no hard and fast rules here and experimentation will be needed to determine
a good value.
The shared pool size is highly application dependent. To
determine the shared pool size that will be needed for a production
system it is generally necessary to first develop the application and run
it on a test system and take some measurements. The test system should be
run with a very large value for the shared pool size to make the
measurements meaningful.
1). OBJECTS STORED IN THE DATABASE
The amount of shared pool that needs to be allocated for objects that are
stored in the database like packages and views is easy to measure. You can
just measure their size directly with the following statement.
This is especially effective because all large pl/sql object should be ’kept’
in the shared pool at all times.
--------------------------------------------------------------------------- */
SELECT SUM(sharable_mem) INTO l_sum_obj
FROM v$db_object_cache;
2). SQL-STATEMENTS
The amount of memory needed to store sql statements in the shared pool is
more difficult to measure because of the needs of dynamic sql. If an
application has no dynamic sql then the amount of memory can simply
be measured after the application has run for a while by just selecting
it out of the shared pool as follows:
--------------------------------------------------------------------------- */
SELECT SUM(sharable_mem) INTO l_sum_sql
FROM v$sqlarea;
/* -------------------------------------------------------------------------3). PER-USER PER-CURSOR MEMORY
You will need to allow around 250 bytes of memory in the shared pool per
concurrent user for each open cursor that the user has whether the cursor
is shared or not. During the peak usage time of the production system, you
can measure this as follows:
--------------------------------------------------------------------------- */
SELECT SUM(250 * users_opening) INTO l_sum_users
FROM v$sqlarea;
/* -------------------------------------------------------------------------4). MTS (MULTI-THREADED SERVER)
If you are using multi-threaded server, then you will need to allow enough
memory for all the shared server users to put their session memory in the
shared pool. This can be measured with the following query:
--------------------------------------------------------------------------- */
SELECT SUM(value) INTO l_sum_mts
FROM v$sesstat s, v$statname n
WHERE s.statistic# = n.statistic#
AND n.name = ’session uga memory max’;
Seite
74 von
82
Ausgabe
5. Januar
1995
Seite -74
Oracle Datenbank Tuning
$NDGLD $*
utl_File.Putf(l_fhd,’\n’);
utl_File.Put_Line(l_fhd,’***************** Used space in SHARED POOL (approximative) *******************’);
utl_File.Put_Line(l_fhd,’ ’);
utl_File.Putf(l_fhd,’Total size for Objects ...... = %s [Bytes]\n’, LPAD(TO_CHAR(l_sum_obj),10,’ ’));
utl_File.Putf(l_fhd,’Total size for SQL-Statements = %s [Bytes]\n’, LPAD(TO_CHAR(l_sum_sql),10,’ ’));
utl_File.Putf(l_fhd,’Total size for User Sessions = %s [Bytes]\n’, LPAD(TO_CHAR(l_sum_users),10,’ ’));
utl_File.Putf(l_fhd,’Total size for MTS .......... = %s [Bytes]\n’, LPAD(TO_CHAR(l_sum_mts),10,’ ’));
utl_File.Putf(l_fhd,’Total size used ............. = %s [Bytes]\n’,
LPAD(TO_CHAR(l_sum_obj+l_sum_sql+l_sum_users+l_sum_mts),10,’ ’));
/* ---------------------------------------------------------------Show memory segments in the shared pool and their state
Sum: Total sum(size) of all memory segments of this type
State: recr
= Flushable segment.
freeabl = This segment is in use and can’t be flushed.
free
= Free segment
perm
= Permanently allocated memory segment
Max: This shows the biggest memory segments of this type.
----------------------------------------------------------------- */
utl_File.Putf(l_fhd,’\n’);
utl_File.Put_Line(l_fhd,’************* Memory Segments and their state in the SHARED POOL **************’);
utl_File.Put_Line(l_fhd,’ ’);
utl_File.Put_Line(l_fhd,’Description
State
Sum (Size)
Max (Size)
Number of’);
utl_File.Put_Line(l_fhd,’
[Bytes]
[Bytes]
Segments’);
utl_File.Put_Line(l_fhd,’---------------- ------------- ----------------- --------------- --------------’);
FOR ksmsp_rec IN ksmsp_cur LOOP
utl_File.Putf(l_fhd,’%s%s%s%s%s\n’, RPAD(ksmsp_rec.ksmchcom,17,’ ’),
RPAD(ksmsp_rec.text_ksmsp,14,’ ’),
LPAD(TO_CHAR(ksmsp_rec.sum_ksmsp),17,’ ’),
LPAD(TO_CHAR(ksmsp_rec.max_ksmsp),16,’ ’),
LPAD(TO_CHAR(ksmsp_rec.count_ksmsp),15,’ ’));
IF (ksmsp_rec.text_ksmsp = ’Flushable’) THEN
l_sum_flush := l_sum_flush + ksmsp_rec.sum_ksmsp;
ELSIF (ksmsp_rec.text_ksmsp = ’InUse’) THEN
l_sum_inuse := l_sum_inuse + ksmsp_rec.sum_ksmsp;
ELSIF (ksmsp_rec.text_ksmsp = ’Free’) THEN
l_sum_free := l_sum_free + ksmsp_rec.sum_ksmsp;
ELSIF (ksmsp_rec.text_ksmsp = ’Permanent’) THEN
l_sum_perm := l_sum_perm + ksmsp_rec.sum_ksmsp;
END IF;
END LOOP;
utl_File.Put_Line(l_fhd,’-------------------------------------------------------------------------------’);
utl_File.Putf(l_fhd,’Total size of flushable memory segments = %s [Bytes]\n’,
LPAD(TO_CHAR(l_sum_flush),10,’ ’));
utl_File.Putf(l_fhd,’Total size of memory segments in use .. = %s [Bytes]\n’,
LPAD(TO_CHAR(l_sum_inuse),10,’ ’));
utl_File.Putf(l_fhd,’Total size of free memory segments .... = %s [Bytes]\n’,
LPAD(TO_CHAR(l_sum_free),10,’ ’));
utl_File.Putf(l_fhd,’Total size of permanent memory segments = %s [Bytes]\n’,
LPAD(TO_CHAR(l_sum_perm),10,’ ’));
/* ------------------------------------------------------------------Number and average size of free memory segments in the shared pool.
The number of free memory segments should be small, but the average
size shuld be big. If these is not the case, the shared pool is
fragmented and we will get the ORA-4031 Error soon.
--------------------------------------------------------------------- */
SELECT COUNT(*), AVG(ksmchsiz)
INTO l_sum_count, l_size
FROM sys.x$ksmsp
WHERE ksmchcom = ’free memory’;
Ausgabe 5. Januar 1995
Seite 75
von-75
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
utl_File.Putf(l_fhd,’Average size of free memory segments .. = %s [Bytes]\n’,
LPAD(TO_CHAR(l_size),10,’ ’));
utl_File.Putf(l_fhd,’Number free memory segments ........... = %s\n’,
LPAD(TO_CHAR(l_sum_count),10,’ ’));
utl_File.Putf(l_fhd,’Average size / number free segments ... = %s\n’,
LPAD(TO_CHAR(l_size/l_sum_count),10,’ ’));
utl_File.Put_Line(l_fhd,’-------------------------------------------------------------------------------’);
utl_File.Putf(l_fhd,’\n’);
/* Large objects in shared pool (size > 10 KBytes) */
utl_File.Put_Line(l_fhd,’************** Big objects in the SHARED POOL (size > 10 KBytes) **************’);
utl_File.Put_Line(l_fhd,’ ’);
utl_File.Put_Line(l_fhd,’Owner
Name
Type
Kept Loads
Bytes’);
utl_File.Put_Line(l_fhd,’-------- ------------------------------ ------------- ---- ----- --------------’);
FOR big_objects_rec IN big_objects_cur LOOP
utl_File.Putf(l_fhd,’%s%s%s%s%s’, RPAD(big_objects_rec.owner,9,’ ’),
RPAD(big_objects_rec.name,31,’ ’),
RPAD(big_objects_rec.type,14,’ ’),
RPAD(big_objects_rec.kept,5,’ ’),
LPAD(TO_CHAR(big_objects_rec.loads),5,’ ’));
utl_File.Putf(l_fhd,’%s\n’, LPAD(TO_CHAR(big_objects_rec.sharable_mem),15,’ ’));
END LOOP;
/* Summary statistic of the shared pool operations */
utl_File.Putf(l_fhd,’\n’);
utl_File.Put_Line(l_fhd,’*************** Summary statistic of the SHARED POOL operations ***************’);
utl_File.Put_Line(l_fhd,’ ’);
utl_File.Put_Line(l_fhd,’LRU-Operations
Flushes Recurrent Segments Transient Segments’);
utl_File.Put_Line(l_fhd,’----------------------- ---------------- ------------------- ------------------’);
FOR ksmlru_rec IN (SELECT * FROM x$kghlu) LOOP
l_count := l_count + 1;
l_kghlu(l_count) := ksmlru_rec;
utl_File.Putf(l_fhd,’%s%s%s%s’, LPAD(TO_CHAR(l_kghlu(l_count).kghluops),23,’
LPAD(TO_CHAR(l_kghlu(l_count).kghlufsh),17,’
LPAD(TO_CHAR(l_kghlu(l_count).kghlurcr),20,’
LPAD(TO_CHAR(l_kghlu(l_count).kghlutrn),19,’
utl_File.Putf(l_fhd,’\n’);
END LOOP;
’),
’),
’),
’));
/* Track allocations in the shared pool that cause */
/* other objects in the shared pool to flushed out */
utl_File.Putf(l_fhd,’\n’);
utl_File.Put_Line(l_fhd,’**** Allocations that cause other objects in the SHARED POOL to flushed out ***’);
utl_File.Put_Line(l_fhd,’ ’);
utl_File.Put_Line(l_fhd,’Command
Comment
Size
Obj flushed’);
utl_File.Put_Line(l_fhd,’-------------------------------- -------------------- ---------- --------------’);
l_count := 0;
l_message := NULL;
FOR flushed_obj_rec IN flushed_obj_cur LOOP
IF (flushed_obj_rec.ksmlrnum > 0) THEN
l_count := l_count + 1;
utl_File.Putf(l_fhd,’%s%s%s%s\n’, RPAD(flushed_obj_rec.ksmlrhon,33,’
RPAD(flushed_obj_rec.ksmlrcom,21,’
LPAD(flushed_obj_rec.ksmlrsiz,10,’
LPAD(flushed_obj_rec.ksmlrnum,15,’
’),
’),
’),
’));
IF (flushed_obj_rec.ksmlrsiz >= 5000 AND flushed_obj_rec.ksmlrsiz < 10000) THEN
l_message := ’ORA-4031 will start soon !’;
l_updt_status := 1;
Seite
76 von
82
Ausgabe
5. Januar
1995
Seite -76
Oracle Datenbank Tuning
$NDGLD $*
ELSIF (flushed_obj_rec.ksmlrsiz > 10000
AND flushed_obj_rec.ksmlrsiz < 20000) THEN
l_message := ’ORA-4031 will be a serious problem soon !’;
l_updt_status := 2;
ELSIF (flushed_obj_rec.ksmlrsiz > 20000) THEN
l_message := ’ORA-4031 will now starting !’;
l_updt_status := 3;
END IF;
END IF;
END LOOP;
IF (l_count = 0) THEN
utl_File.Put_Line(l_fhd,’No objects have been flushed out ... wonderful !’);
l_updt_status := 0;
END IF;
IF (l_message IS NOT NULL) THEN
utl_File.Putf(l_fhd,’%s\n’, l_message);
END IF;
/* Tuning of SHARED_POOL_RESERVED_SIZE */
utl_File.Putf(l_fhd,’\n’);
utl_File.Put_Line(l_fhd,’******************** Tuning of SHARED_POOL_RESERVED_SIZE **********************’);
utl_File.Put_Line(l_fhd,’ ’);
utl_File.Put_Line(l_fhd,’
FreeSpace FreeSegm
UsedSpace UsedSegm Requests Flushes ORA-4031’);
utl_File.Put_Line(l_fhd,’
[Bytes]
Counts
[Bytes]
Counts
Counts
Counts Counts’);
utl_File.Put_Line(l_fhd,’-------------- --------- -------------- --------- ---------- --------- --------’);
FOR spr_rec IN spr_cur LOOP
utl_File.Putf(l_fhd,’%s%s%s%s%s’, LPAD(spr_rec.free_space,14,’ ’),
LPAD(spr_rec.free_count,10,’ ’),
LPAD(spr_rec.used_space,15,’ ’),
LPAD(spr_rec.used_count,10,’ ’),
LPAD(spr_rec.requests,11,’ ’));
utl_File.Putf(l_fhd,’%s%s\n’, LPAD(spr_rec.request_misses,10,’ ’),
LPAD(spr_rec.request_failures,9,’ ’));
utl_File.Putf(l_fhd,’\n’);
IF (spr_rec.request_failures > 0) THEN
l_message := ’Your System has ORA-4031 Errors !’;
l_updt_status := 3;
utl_File.Putf(l_fhd,’%s\n’, l_message);
END IF;
IF (spr_rec.last_failure_size > l_sprma
AND spr_rec.max_free_size < l_sprma
AND spr_rec.free_space < l_sprma) THEN
l_message := ’Your SHARED_POOL_RESERVED_SIZE seems to bee too small !’;
l_updt_status := 2;
utl_File.Putf(l_fhd,’%s\n’, l_message);
END IF;
IF (spr_rec.request_misses = 0
AND spr_rec.free_space > (l_sprma/2)) THEN
l_message := ’Your SHARED_POOL_RESERVED_SIZE seems to be too big’;
utl_File.Putf(l_fhd,’%s\n’, l_message);
END IF;
IF (spr_rec.request_failures > 0
AND spr_rec.last_failure_size < l_sprma) THEN
l_message := ’Your SHARED_POOL_SIZE is really too small !’;
l_updt_status := 3;
utl_File.Putf(l_fhd,’%s\n’, l_message);
END IF;
END LOOP;
utl_File.Putf(l_fhd,’\n’);
utl_file.FClose(l_fhd);
Ausgabe 5. Januar 1995
Seite 77
von-77
82
Seite
Oracle Datenbank Tuning
$NDGLD $*
/* Logging in table eventlog */
BEGIN
UPDATE eventlog SET status = l_updt_status
WHERE event = ’shared_pool’;
COMMIT;
EXCEPTION
WHEN NO_DATA_FOUND THEN
RAISE_APPLICATION_ERROR(-20001, ’Check_Shared_Pool(): UPDATE failed for EVENTLOG’);
ROLLBACK;
END;
EXCEPTION
WHEN OTHERS THEN
l_message := ’Check_Shared_Pool(): ’ || SUBSTR(SQLERRM,1,100);
RAISE_APPLICATION_ERROR(-20000,l_message);
ROLLBACK;
END;
Seite
78 von
82
Ausgabe
5. Januar
1995
Seite -78
$NDGLD $*
Oracle Datenbank Tuning
Output der
Überwachung
Output der Stored Procedure check_shared_pool():
-------------------------- Shared Pool statistics ----------------------------27.05.1997 20:56 - 12.06.1997 08:49
T18M7S.WORLD
------------------------------------------------------------------------------*********************** Current Size of SHARED POOL ***************************
Current size of SHARED_POOL_SIZE ............. =
Current size of SHARED_POOL_RESERVED_SIZE .... =
Current size of SHARED_POOL_RESERVED_MIN_ALLOC =
41943040 [Bytes]
8000000 [Bytes]
1000000 [Bytes]
***************** Used space in SHARED POOL (approximative) *******************
Total
Total
Total
Total
Total
size
size
size
size
size
for Objects ......
for SQL-Statements
for User Sessions
for MTS ..........
used .............
=
=
=
=
=
1885707
15009721
4250
467992
17367670
[Bytes]
[Bytes]
[Bytes]
[Bytes]
[Bytes]
************* Memory Segments and their state in the SHARED POOL **************
Description
Sum (Size)
Max (Size)
Number of
[Bytes]
[Bytes]
Segments
---------------- ------------- ----------------- --------------- -------------character set m InUse
32896
15152
9
dictionary cach InUse
1376624
4256
814
fixed allocatio Flushable
240
40
6
free memory
Free
6258376
6216568
743
free memory
R-free
8000000
8000000
1
KGL handles
Flushable
1596184
960
5272
kxfp subheap
Flushable
584
584
1
kzull
InUse
304
64
6
library cache
InUse
2545592
584
8643
library cache
Flushable
2210656
584
4708
multiblock rea InUse
560
280
2
permanent memor Permanent
16420872
16420872
1
PLS cca hp desc InUse
168
168
1
PLS non-lib hp Flushable
2104
2104
1
PL/SQL DIANA
InUse
1044264
2112
626
PL/SQL DIANA
Flushable
114392
2104
72
PL/SQL MPCODE
InUse
449968
8400
179
PL/SQL MPCODE
Flushable
97344
2104
90
reserved stoppe R-freea
48
24
2
row cache lru
Flushable
1944
64
40
session param v InUse
31440
2096
15
sql area
InUse
7230464
4256
2333
sql area
Flushable
2559104
4264
808
table definiti Flushable
64
64
1
trigger defini Flushable
1080
1080
1
------------------------------------------------------------------------------Total size of flushable memory segments =
6583696 [Bytes]
Total size of memory segments in use .. = 12712280 [Bytes]
Total size of free memory segments .... =
6258376 [Bytes]
Total size of permanent memory segments = 16420872 [Bytes]
Average size of free memory segments .. =
19182 [Bytes]
Number free memory segments ........... =
743
Average size / number free segments ... = 25,8169582
-------------------------------------------------------------------------------
Ausgabe 5. Januar 1995
State
Seite 79
von-79
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
************** Big objects in the SHARED POOL (size > 10 KBytes) **************
Owner
-------SYS
SYS
SIPP
SIPP
SYS
SYS
SYS
SYS
SYS
SYS
SYS
SYS
SYS
PUBLIC
SYS
SIPP
SYS
SIPP
Name
-----------------------------CHECK_SHARED_POOL
STANDARD
CHECK_ACCM_MAX_TOTAL
SUM_RELOAD
STANDARD
INDREORG
DBMS_IREFRESH
DBMS_SYS_SQL
DBMS_SHARED_POOL
V$SQLAREA
DBMS_UTILITY
DBMS_IREFRESH
V$SHARED_POOL_RESERVED
SSCORDER
X$KGHLU
SEQ_ORDER_REQ_ORD_ID
X$KSMLRU
SEQ_ESRLOG_LOG_ID
Type
------------PROCEDURE
PACKAGE
PROCEDURE
PROCEDURE
PACKAGE BODY
PROCEDURE
PACKAGE BODY
PACKAGE BODY
PACKAGE
VIEW
PACKAGE
PACKAGE
VIEW
SYNONYM
TABLE
SEQUENCE
TABLE
SEQUENCE
Kept Loads
Bytes
---- ----- -------------NO
1
140981
NO
1
120628
NO
3
34577
NO
4
34559
NO
1
28700
NO
1
25028
NO
1
22713
NO
1
22448
NO
1
21956
NO
1
19741
NO
1
19664
NO
1
19057
NO
1
17538
NO
1
17351
NO
1
17163
NO
21
17105
NO
1
17092
NO
21
16974
*************** Summary statistic of the SHARED POOL operations ***************
LRU-Operations
Flushes Recurrent Segments Transient Segments
----------------------- ---------------- ------------------- -----------------771534
0
2652
5611
**** Allocations that cause other objects in the SHARED POOL to flushed out ***
Command
Comment
Size
Obj flushed
-------------------------------- -------------------- ---------- -------------No objects have been flushed out ... wonderful !
******************** Tuning of SHARED_POOL_RESERVED_SIZE **********************
FreeSpace FreeSegm
UsedSpace UsedSegm Requests Flushes ORA-4031
[Bytes]
Counts
[Bytes]
Counts
Counts
Counts Counts
-------------- --------- -------------- --------- ---------- --------- -------8000000
1
0
0
0
0
0
Your SHARED_POOL_RESERVED_SIZE seems to be too big
Seite
80 von
82
Ausgabe
5. Januar
1995
Seite -80
$NDGLD $*
Oracle Datenbank Tuning
5.
Locking Verhalten
5.1
Row Locks
Row Share Table
Locks (RS)
Ein Row Share Table Lock zeigt, dass gewisse Rows in der Table für
einen UPDATE gelockt worden sind. Der RS-Lock ist der am wenigsten
restriktive Mode, erlaubt also die grösste Concurreny.
Data Manipulation
Transaction
Beispiele
SELECT person_nr FROM mitglied FOR UPDATE OF pult_nr;
LOCK TABLE mitglied IN EXCLUSIVE MODE;
Row Exclusive Table
Locks (RX)
Ein Row Exclusive Table Lock zeigt, dass gewisse Rows geändert wurden, aber noch kein COMMIT oder ROLLBACK erfolgt ist. Der RX-Lock
ist restriktiver als der RS-Lock.
Beispiele
UPDATE mitglied SET name = ’Zahn’ WHERE name = ’Eiger’;
INSERT INTO table .....;
DELETE FROM table .....;
LOCK TABLE mitglied in ROW EXCLUSIVE MODE;
Ausgabe 5. Januar 1995
Seite 81
von-81
82
Seite
$NDGLD $*
Oracle Datenbank Tuning
5.2
Table Locks
Table Locks werden ausschliesslich durch explizite Locking Commands
ausgelöst und sollten grundsätzlich vermieden werden, da sie das Default
Lockingverhalten von Oracle ausschalten.
Oracle kennt folgende Table Locks:
• Share Table Locks (S)
LOCK TABLE mitglied IN SHARE MODE;
• Share Row Exclusive Table Locks (SRX)
LOCK TABLE mitglied IN SHARE ROW EXCLUSIVE MODE;
• Exclusive Table Locks (X)
LOCK TABLE mitglied IN EXCLUSIVE MODE;
5.3
Lock Manager Lock Types
Im Enterprise Manager Lock Manager werden folgende Lock Types benutzt:
MR
Media Recovery
IS
Instance State
RT
Redo Thread
FS
File Set
UN
User Name
IR
Instance Recovery
TX
Transaction
ST
Disk Space Transaction
TM
Data Manipulation
TS
Temp Segment
UL
PL/SQL User Lock
IV
Library cache invalidation
DX
Distributed Action
LS
Log Start/Switch
CF
Control File
RW
Row Wait
SQ
Sequence Number
TE
Extend Table
TT
Temp Table
Lock Manager: Lock Types
Seite
82 von
82
Ausgabe
5. Januar
1995
Seite -82