July 21, 2011
Have you ever wondered about the V$OPEN_CURSOR view, the SESSION_CACHED_CURSORS parameter, and the two session-level statistics “session cursor cache count” and “session cursor cache hits”? I did after reading from two different sources that stated essentially that a larger shared pool would be required when the value for the SESSION_CACHED_CURSORS parameter is increased. The shared pool? – but the session’s cursors are cached in the UGA, which is typically in the PGA (unless shared server is used), and as far as I am aware the PGA is not in the shared pool.
So, what do we (think that we) know about the session cached cursors?
- The per session limit for the number of cached cursors is specified by the SESSION_CACHED_CURSORS parameter.
- Prior to the release of the Oracle Database 9.2.0.5 patchset the OPEN_CURSORS parameter controlled the number of cached cursors for PL/SQL execution, and as of 9.2.0.5 the SESSION_CACHED_CURSORS parameter controls the number of PL/SQL cursors that are cached per session (source).
- In recent Oracle Database release versions the SESSION_CACHED_CURSORS parameter defaults to a value of 50.
- Assuming that the SESSION_CACHED_CURSORS parameter is set to a value greater than 0, a SQL statement that is executed three times in the same session will be added to that session’s cursor cache and will remain open, even when explicitly closed by the session.
- Starting in Oracle Database 11.1, if a 10046 trace is enabled for a session and a cursor is closed, a CLOSE # line will be written to the 10046 trace file, with the type= parameter indicating whether or not the cursor was added to the session cursor cache (and was thus actually not closed).
- In V$MYSTAT, the session-level “session cursor cache count” statistic indicates the number of cursors that are cached for the current session, while the “session cursor cached hits” statistic indicates the number of times the current session has accessed those cached cursors.
- V$OPEN_CURSOR shows the cursors that are currently open for all connected sessions. Those cursors that are in the session cursor cache will also appear in V$OPEN_CURSOR.
The above bullet points are probably common knowledge, but is it all true?
Let’s check the Oracle Database 11.2 documentation for V$OPEN_CURSOR:
“V$OPEN_CURSOR lists cursors that each user session currently has opened and parsed, or cached.”
Nothing out of the ordinary with the above quote – it seems to be well aligned with the last of the above bullet points.
Let’s check the Oracle Database 11.2 documentation for SESSION_CACHED_CURSORS:
Default value: 50
“SESSION_CACHED_CURSORS specifies the number of session cursors to cache. Repeated parse calls of the same SQL (including recursive SQL) or PL/SQL statement cause the session cursor for that statement to be moved into the session cursor cache. Subsequent parse calls will find the cursor in the cache and do not need to reopen the cursor. Oracle uses a least recently used algorithm to remove entries in the session cursor cache to make room for new entries when needed.”
The above confirms bullet points one and three that were listed above, and almost bullet point two.
Let’s check the behavior changes listed for Oracle Database 11.2 in the documentation library:
“Starting with Oracle Database 10g Release 1 (10.1), the number of cached cursors is determined by the SESSION_CACHED_CURSORS initialization parameter. In previous Oracle Database releases, the number of SQL cursors cached by PL/SQL was determined by the OPEN_CURSORS initialization parameter.”
Although the above quote states Oracle Database 10.1 rather than 9.2.0.5, the quote confirms the change mentioned in the second of the above bullet points.
Let’s take a look at part of the Performance Tuning Guide from the 11.2 documentation set:
“Oracle checks the library cache to determine whether more than three parse requests have been issued on a given statement. If so, then Oracle assumes that the session cursor associated with the statement should be cached and moves the cursor into the session cursor cache. Subsequent requests to parse that SQL statement by the same session then find the cursor in the session cursor cache.”
Although the above states that a SQL statement must be parsed more than three times before it is added to the session cursor cache rather than executed three times, the quote basically confirms the fourth of the above bullet points.
Checking the statistic descriptions for Oracle Database 11.2:
“session cursor cache count: Total number of cursors cached. This statistic is incremented only if SESSION_CACHED_CURSORS > 0. This statistic is the most useful in V$SESSTAT. If the value for this statistic in V$SESSTAT is close to the setting of the SESSION_CACHED_CURSORS parameter, the value of the parameter should be increased.
session cursor cache hits: Number of hits in the session cursor cache. A hit means that the SQL statement did not have to be reparsed. Subtract this statistic from “parse count (total)” to determine the real number of parses that occurred.”
The above quote seems to confirm the second to the last of the above bullet points.
Everyone happy, or should we test?
We need a table for a bit of experimentation, and we might as well flush the shared pool too:
CREATE TABLE T4 AS
SELECT
*
FROM
ALL_OBJECTS
WHERE
ROWNUM<=10;
EXEC DBMS_STATS.GATHER_TABLE_STATS(OWNNAME=>USER,TABNAME=>'T4')
ALTER SYSTEM FLUSH SHARED_POOL;
Now we need some sort of an elegant way to test the bulleted items that are listed at the start of this article. Of course, a VBS script that is executed on the poor Windows client computer that is sitting under the finger smudged iPad is the perfect way to run a test. In the script below replace MyUsername with a valid username for your database, replace MyPassword with the password for that user, and replace MyDatabase with the database SID as it appears in the TNSNAMES.ORA file. The script performs the following actions:
- Connects to the database
- Determines the SID and SERIAL# for the session and writes those values to the C:\SessionCursorCache.txt text file.
- Enables a 10046 trace at level 1 (minimal detail with no wait events)
- Writes the session’s current values for the “session cursor cache hits” and “session cursor cache count” to the text file.
- Writes to the text file all SQL statements from V$OPEN_CURSOR where the SID matches the SID for the current session.
- Opens two cursors for the SQL statement “SELECT * FROM T4”
- Writes the session’s current values for the “session cursor cache hits” and “session cursor cache count” to the text file. This should indicate if opening the cursors in step 6 triggered a hit on the cached cursors.
- Retrieves all rows from the two cursors and explicitly closes those cursors (potentially alternating between fetches from the two cursors if table T4 contained more than 1,000 rows).
- Repeat steps 4 through 8 two more times to see how repeated executions affect the statistics and the cursors that are cached.
- Disconnects from the database.
The script follows (script download, save with .vbs extension SessionCachedCursorsTest.vbs):
Dim strSQL
Dim strSQL2
Dim strSQL3
Dim strUsername
Dim strPassword
Dim strDatabase
Dim snpData
Dim snpData2
Dim snpDataCursorCache
Dim snpDataOpenSQL
Dim snpDataSID
Dim dbDatabase
Dim lngCnt
Dim intPass
Dim objFSO
Dim objFile
Set objFSO = CreateObject("Scripting.FileSystemObject")
Set objFile=objFSO.CreateTextFile("C:\SessionCursorCache.txt", True)
Set snpData = CreateObject("ADODB.Recordset")
Set snpData2 = CreateObject("ADODB.Recordset")
Set snpDataCursorCache = CreateObject("ADODB.Recordset")
Set snpDataOpenSQL = CreateObject("ADODB.Recordset")
Set snpDataSID = CreateObject("ADODB.Recordset")
Set dbDatabase = CreateObject("ADODB.Connection")
strUsername = "MyUsername"
strPassword = "MyPassword"
strDatabase = "MyDatabase"
dbDatabase.ConnectionString = "Provider=OraOLEDB.Oracle;Data Source=" & strDatabase & ";User ID=" & strUsername & ";Password=" & strPassword & ";FetchSize=5000;"
dbDatabase.Open
strSQL = "SELECT" & VBCrLf
strSQL = strSQL & " S.SID," & VBCrLf
strSQL = strSQL & " S.SERIAL# SERIALN" & VBCrLf
strSQL = strSQL & "FROM" & VBCrLf
strSQL = strSQL & " V$SESSION S," & VBCrLf
strSQL = strSQL & " (SELECT" & VBCrLf
strSQL = strSQL & " SID" & VBCrLf
strSQL = strSQL & " FROM" & VBCrLf
strSQL = strSQL & " V$MYSTAT" & VBCrLf
strSQL = strSQL & " WHERE" & VBCrLf
strSQL = strSQL & " ROWNUM=1) M" & VBCrLf
strSQL = strSQL & "WHERE" & VBCrLf
strSQL = strSQL & " S.SID=M.SID" & VBCrLf
snpDataSID.Open strSQL, dbDatabase
If snpDataSID.State = 1 Then
Do While Not(snpDataSID.EOF)
objFile.Write "SID: " & snpDataSID("sid") & ", SERIAL#: " & snpDataSID("serialn") & VBCrLf
snpDataSID.MoveNext
Loop
snpDataSID.Close
objFile.Write " " & VBCrLf
End If
dbDatabase.Execute "ALTER SESSION SET TRACEFILE_IDENTIFIER = 'SessionCursorCache'"
dbDatabase.Execute "ALTER SESSION SET EVENTS '10046 TRACE NAME CONTEXT FOREVER, LEVEL 1'"
strSQL2 = "SELECT" & VBCrLf
strSQL2 = strSQL2 & " SN.NAME STAT_NAME," & VBCrLf
strSQL2 = strSQL2 & " M.VALUE" & VBCrLf
strSQL2 = strSQL2 & "FROM" & VBCrLf
strSQL2 = strSQL2 & " V$MYSTAT M," & VBCrLf
strSQL2 = strSQL2 & " V$STATNAME SN" & VBCrLf
strSQL2 = strSQL2 & "WHERE" & VBCrLf
strSQL2 = strSQL2 & " SN.NAME IN ('session cursor cache hits','session cursor cache count')" & VBCrLf
strSQL2 = strSQL2 & " AND SN.STATISTIC#=M.STATISTIC#" & VBCrLf
strSQL2 = strSQL2 & "ORDER BY" & VBCrLf
strSQL2 = strSQL2 & " SN.NAME"
strSQL3 = "SELECT" & VBCrLf
strSQL3 = strSQL3 & " SQL_TEXT" & VBCrLf
strSQL3 = strSQL3 & "FROM" & VBCrLf
strSQL3 = strSQL3 & " V$OPEN_CURSOR" & VBCrLf
strSQL3 = strSQL3 & "WHERE" & VBCrLf
strSQL3 = strSQL3 & " SID=" & VBCrLf
strSQL3 = strSQL3 & " (SELECT" & VBCrLf
strSQL3 = strSQL3 & " SID" & VBCrLf
strSQL3 = strSQL3 & " FROM" & VBCrLf
strSQL3 = strSQL3 & " V$MYSTAT" & VBCrLf
strSQL3 = strSQL3 & " WHERE" & VBCrLf
strSQL3 = strSQL3 & " ROWNUM=1)" & VBCrLf
strSQL3 = strSQL3 & "ORDER BY" & VBCrLf
strSQL3 = strSQL3 & " SQL_TEXT"
For intPass = 1 to 3
snpDataCursorCache.Open strSQL2, dbDatabase
If snpDataCursorCache.State = 1 Then
objFile.Write "Before Pass #" & intPass & VBCrLf
Do While Not(snpDataCursorCache.EOF)
objFile.Write snpDataCursorCache("stat_name") & " " & snpDataCursorCache("value") & VBCrLf
snpDataCursorCache.MoveNext
Loop
snpDataCursorCache.Close
objFile.Write " " & VBCrLf
End If
snpDataOpenSQL.Open strSQL3, dbDatabase
lngCnt = 0
If snpDataOpenSQL.State = 1 Then
Do While Not(snpDataOpenSQL.EOF)
lngCnt = lngCnt + 1
objFile.Write lngCnt & " " & snpDataOpenSQL("sql_text") & VBCrLf
snpDataOpenSQL.MoveNext
Loop
snpDataOpenSQL.Close
objFile.Write " " & VBCrLf
End If
strSQL = "SELECT * FROM T4"
snpData.Open strSQL, dbDatabase
snpData2.Open strSQL, dbDatabase
snpDataCursorCache.Open strSQL2, dbDatabase
If snpDataCursorCache.State = 1 Then
objFile.Write "After Open Pass #" & intPass & VBCrLf
Do While Not(snpDataCursorCache.EOF)
objFile.Write snpDataCursorCache("stat_name") & " " & snpDataCursorCache("value") & VBCrLf
snpDataCursorCache.MoveNext
Loop
snpDataCursorCache.Close
objFile.Write " " & VBCrLf
End If
lngCnt = 0
If snpData.State = 1 Then
Do While Not(snpData.EOF)
Do While (lngCnt < 1000) and Not(snpData.EOF)
lngCnt = lngCnt + 1
snpData.MoveNext
Loop
lngCnt = 0
Do While (lngCnt < 1000) and Not(snpData2.EOF)
lngCnt = lngCnt + 1
snpData2.MoveNext
Loop
lngCnt = 0
Loop
End If
snpData.Close
snpData2.Close
Next
dbDatabase.Execute "ALTER SESSION SET EVENTS '10046 TRACE NAME CONTEXT OFF'"
objFile.Close
Set snpData = Nothing
Set snpData2 = Nothing
Set snpDataCursorCache = Nothing
Set snpDataOpenSQL = Nothing
Set snpDataSID = Nothing
dbDatabase.Close
Set dbDatabase = Nothing
Set objFile = Nothing
Set objFSO = Nothing
If you save the script as SessionCachedCursorsTest.vbs you can execute it from a Windows command line with the following command:
CSCRIPT SessionCachedCursorsTest.vbs
The script will generate a file named C:\SessionCursorCache.txt (if User Access Control is enabled on the Windows computer, you will need to modify the script to create the file in a different location). What might you find in the generated file? This is what I saw after the script ran shortly after flushing the shared pool (64 bit Oracle Database 11.2.0.2):
SID: 66, SERIAL#: 7863
Before Pass #1
session cursor cache count 49
session cursor cache hits 240
1 SELECT SQL_TEXT FROM V$OPEN_CURSOR WHERE SID=
2 insert into sys.aud$( sessionid,entryid,statement,ntimestamp
3 select /*+ rule */ bucket, endpoint, col#, epvalue from hist
4 select /*+ rule */ bucket, endpoint, col#, epvalue from hist
5 select /*+ rule */ bucket_cnt, row_cnt, cache_cnt, null_cnt,
6 select /*+ rule */ bucket_cnt, row_cnt, cache_cnt, null_cnt,
7 select /*+ rule */ bucket_cnt, row_cnt, cache_cnt, null_cnt,
8 select col#, grantee#, privilege#,max(mod(nvl(option$,0),2))
9 select col#, grantee#, privilege#,max(mod(nvl(option$,0),2))
10 select col#, grantee#, privilege#,max(mod(nvl(option$,0),2))
11 select col#,intcol#,charsetid,charsetform from col$ where ob
12 select col#,intcol#,charsetid,charsetform from col$ where ob
13 select col#,intcol#,charsetid,charsetform from col$ where ob
14 select col#,intcol#,ntab# from ntab$ where obj#=:1 order by
15 select col#,intcol#,ntab# from ntab$ where obj#=:1 order by
16 select col#,intcol#,ntab# from ntab$ where obj#=:1 order by
17 select col#,intcol#,reftyp,stabid,expctoid from refcon$ wher
18 select col#,intcol#,reftyp,stabid,expctoid from refcon$ wher
19 select col#,intcol#,reftyp,stabid,expctoid from refcon$ wher
20 select col#,intcol#,toid,version#,packed,intcols,intcol#s,fl
21 select col#,intcol#,toid,version#,packed,intcols,intcol#s,fl
22 select col#,intcol#,toid,version#,packed,intcols,intcol#s,fl
23 select cols,audit$,textlength,intcols,property,flags,rowid f
24 select cols,audit$,textlength,intcols,property,flags,rowid f
25 select con#,obj#,rcon#,enabled,nvl(defer,0),spare2,spare3 fr
26 select con#,obj#,rcon#,enabled,nvl(defer,0),spare2,spare3 fr
27 select con#,type#,condlength,intcols,robj#,rcon#,match#,refa
28 select con#,type#,condlength,intcols,robj#,rcon#,match#,refa
29 select distinct(-privilege#),nvl(option$,0) from sysauth$ wh
30 select grantee#,privilege#,nvl(col#,0),max(mod(nvl(option$,0
31 select grantee#,privilege#,nvl(col#,0),max(mod(nvl(option$,0
32 select grantee#,privilege#,nvl(col#,0),max(mod(nvl(option$,0
33 select intcol#, toid, version#, intcols, intcol#s, flags, sy
34 select intcol#, toid, version#, intcols, intcol#s, flags, sy
35 select intcol#, toid, version#, intcols, intcol#s, flags, sy
36 select intcol#,type,flags,lobcol,objcol,extracol,schemaoid,
37 select intcol#,type,flags,lobcol,objcol,extracol,schemaoid,
38 select intcol#,type,flags,lobcol,objcol,extracol,schemaoid,
39 select l.col#, l.intcol#, l.lobj#, l.ind#, l.ts#, l.file#, l
40 select name,intcol#,segcol#,type#,length,nvl(precision#,0),d
41 select name,intcol#,segcol#,type#,length,nvl(precision#,0),d
42 select node,owner,name from syn$ where obj#=:1
43 select node,owner,name from syn$ where obj#=:1
44 select obj#,type#,ctime,mtime,stime, status, dataobj#, flags
45 select obj#,type#,ctime,mtime,stime, status, dataobj#, flags
46 select order#,columns,types from access$ where d_obj#=:1
47 select owner#,name,namespace,remoteowner,linkname,p_timestam
48 select rowcnt, blkcnt, empcnt, avgspc, chncnt, avgrln, analy
49 select text from view$ where rowid=:1
50 select text from view$ where rowid=:1
51 select timestamp, flags from fixed_obj$ where obj#=:1
52 select timestamp, flags from fixed_obj$ where obj#=:1
After Open Pass #1
session cursor cache count 49
session cursor cache hits 279
Before Pass #2
session cursor cache count 49
session cursor cache hits 279
1 SELECT SN.NAME STAT_NAME, M.VALUE FROM V$MYSTAT M,
2 SELECT SQL_TEXT FROM V$OPEN_CURSOR WHERE SID=
3 insert into sys.aud$( sessionid,entryid,statement,ntimestamp
4 select /*+ rule */ bucket, endpoint, col#, epvalue from hist
5 select /*+ rule */ bucket_cnt, row_cnt, cache_cnt, null_cnt,
6 select /*+ rule */ bucket_cnt, row_cnt, cache_cnt, null_cnt,
7 select /*+ rule */ bucket_cnt, row_cnt, cache_cnt, null_cnt,
8 select col#, grantee#, privilege#,max(mod(nvl(option$,0),2))
9 select col#, grantee#, privilege#,max(mod(nvl(option$,0),2))
10 select col#, grantee#, privilege#,max(mod(nvl(option$,0),2))
11 select col#,intcol#,charsetid,charsetform from col$ where ob
12 select col#,intcol#,charsetid,charsetform from col$ where ob
13 select col#,intcol#,charsetid,charsetform from col$ where ob
14 select col#,intcol#,ntab# from ntab$ where obj#=:1 order by
15 select col#,intcol#,ntab# from ntab$ where obj#=:1 order by
16 select col#,intcol#,reftyp,stabid,expctoid from refcon$ wher
17 select col#,intcol#,reftyp,stabid,expctoid from refcon$ wher
18 select col#,intcol#,toid,version#,packed,intcols,intcol#s,fl
19 select col#,intcol#,toid,version#,packed,intcols,intcol#s,fl
20 select cols,audit$,textlength,intcols,property,flags,rowid f
21 select cols,audit$,textlength,intcols,property,flags,rowid f
22 select con#,obj#,rcon#,enabled,nvl(defer,0),spare2,spare3 fr
23 select con#,obj#,rcon#,enabled,nvl(defer,0),spare2,spare3 fr
24 select con#,obj#,rcon#,enabled,nvl(defer,0),spare2,spare3 fr
25 select con#,type#,condlength,intcols,robj#,rcon#,match#,refa
26 select con#,type#,condlength,intcols,robj#,rcon#,match#,refa
27 select con#,type#,condlength,intcols,robj#,rcon#,match#,refa
28 select distinct(-privilege#),nvl(option$,0) from sysauth$ wh
29 select grantee#,privilege#,nvl(col#,0),max(mod(nvl(option$,0
30 select grantee#,privilege#,nvl(col#,0),max(mod(nvl(option$,0
31 select grantee#,privilege#,nvl(col#,0),max(mod(nvl(option$,0
32 select i.obj#,i.ts#,i.file#,i.block#,i.intcols,i.type#,i.fla
33 select intcol#, toid, version#, intcols, intcol#s, flags, sy
34 select intcol#, toid, version#, intcols, intcol#s, flags, sy
35 select intcol#,nvl(pos#,0),col#,nvl(spare1,0) from ccol$ whe
36 select intcol#,type,flags,lobcol,objcol,extracol,schemaoid,
37 select intcol#,type,flags,lobcol,objcol,extracol,schemaoid,
38 select intcol#,type,flags,lobcol,objcol,extracol,schemaoid,
39 select name,intcol#,segcol#,type#,length,nvl(precision#,0),d
40 select name,intcol#,segcol#,type#,length,nvl(precision#,0),d
41 select name,intcol#,segcol#,type#,length,nvl(precision#,0),d
42 select node,owner,name from syn$ where obj#=:1
43 select node,owner,name from syn$ where obj#=:1
44 select obj#,type#,ctime,mtime,stime, status, dataobj#, flags
45 select obj#,type#,ctime,mtime,stime, status, dataobj#, flags
46 select obj#,type#,ctime,mtime,stime, status, dataobj#, flags
47 select order#,columns,types from access$ where d_obj#=:1
48 select owner#,name,namespace,remoteowner,linkname,p_timestam
49 select t.ts#,t.file#,t.block#,nvl(t.bobj#,0),nvl(t.tab#,0),t
50 select text from view$ where rowid=:1
51 select timestamp, flags from fixed_obj$ where obj#=:1
52 select type#,blocks,extents,minexts,maxexts,extsize,extpct,u
After Open Pass #2
session cursor cache count 50
session cursor cache hits 280
Before Pass #3
session cursor cache count 50
session cursor cache hits 281
1 SELECT SN.NAME STAT_NAME, M.VALUE FROM V$MYSTAT M,
2 SELECT SQL_TEXT FROM V$OPEN_CURSOR WHERE SID=
3 SELECT * FROM T4
4 SELECT * FROM T4
5 insert into sys.aud$( sessionid,entryid,statement,ntimestamp
6 select /*+ rule */ bucket, endpoint, col#, epvalue from hist
7 select /*+ rule */ bucket_cnt, row_cnt, cache_cnt, null_cnt,
8 select /*+ rule */ bucket_cnt, row_cnt, cache_cnt, null_cnt,
9 select /*+ rule */ bucket_cnt, row_cnt, cache_cnt, null_cnt,
10 select col#, grantee#, privilege#,max(mod(nvl(option$,0),2))
11 select col#, grantee#, privilege#,max(mod(nvl(option$,0),2))
12 select col#, grantee#, privilege#,max(mod(nvl(option$,0),2))
13 select col#,intcol#,charsetid,charsetform from col$ where ob
14 select col#,intcol#,charsetid,charsetform from col$ where ob
15 select col#,intcol#,ntab# from ntab$ where obj#=:1 order by
16 select col#,intcol#,ntab# from ntab$ where obj#=:1 order by
17 select col#,intcol#,reftyp,stabid,expctoid from refcon$ wher
18 select col#,intcol#,reftyp,stabid,expctoid from refcon$ wher
19 select col#,intcol#,toid,version#,packed,intcols,intcol#s,fl
20 select col#,intcol#,toid,version#,packed,intcols,intcol#s,fl
21 select cols,audit$,textlength,intcols,property,flags,rowid f
22 select cols,audit$,textlength,intcols,property,flags,rowid f
23 select con#,obj#,rcon#,enabled,nvl(defer,0),spare2,spare3 fr
24 select con#,obj#,rcon#,enabled,nvl(defer,0),spare2,spare3 fr
25 select con#,obj#,rcon#,enabled,nvl(defer,0),spare2,spare3 fr
26 select con#,type#,condlength,intcols,robj#,rcon#,match#,refa
27 select con#,type#,condlength,intcols,robj#,rcon#,match#,refa
28 select con#,type#,condlength,intcols,robj#,rcon#,match#,refa
29 select distinct(-privilege#),nvl(option$,0) from sysauth$ wh
30 select grantee#,privilege#,nvl(col#,0),max(mod(nvl(option$,0
31 select grantee#,privilege#,nvl(col#,0),max(mod(nvl(option$,0
32 select grantee#,privilege#,nvl(col#,0),max(mod(nvl(option$,0
33 select i.obj#,i.ts#,i.file#,i.block#,i.intcols,i.type#,i.fla
34 select intcol#, toid, version#, intcols, intcol#s, flags, sy
35 select intcol#, toid, version#, intcols, intcol#s, flags, sy
36 select intcol#,nvl(pos#,0),col#,nvl(spare1,0) from ccol$ whe
37 select intcol#,type,flags,lobcol,objcol,extracol,schemaoid,
38 select intcol#,type,flags,lobcol,objcol,extracol,schemaoid,
39 select name,intcol#,segcol#,type#,length,nvl(precision#,0),d
40 select name,intcol#,segcol#,type#,length,nvl(precision#,0),d
41 select name,intcol#,segcol#,type#,length,nvl(precision#,0),d
42 select node,owner,name from syn$ where obj#=:1
43 select node,owner,name from syn$ where obj#=:1
44 select obj#,type#,ctime,mtime,stime, status, dataobj#, flags
45 select obj#,type#,ctime,mtime,stime, status, dataobj#, flags
46 select obj#,type#,ctime,mtime,stime, status, dataobj#, flags
47 select order#,columns,types from access$ where d_obj#=:1
48 select owner#,name,namespace,remoteowner,linkname,p_timestam
49 select t.ts#,t.file#,t.block#,nvl(t.bobj#,0),nvl(t.tab#,0),t
50 select text from view$ where rowid=:1
51 select timestamp, flags from fixed_obj$ where obj#=:1
52 select type#,blocks,extents,minexts,maxexts,extsize,extpct,u
After Open Pass #3
session cursor cache count 50
session cursor cache hits 284
Reading the above, before our newly created session even selected from table T4, the statistics indicate that somehow there were already 49 cursors in that session’s cursor cache (many of the SQL statements appeared multiple times) and there were already 240 hits on those cached cursors. Did this session really perform more than 387 (49*3 + 240) executions of SQL statements before our first selection from T4, or does V$OPEN_CURSOR perform differently than described in the documentation? By the time the two cursors selecting from T4 were opened, the number of cursor cache hits jumped by 39. After the two cursors were closed (Before Pass #2) we are able to confirm that the select from table T4 was not added to the session cursor cache and there were still 49 cursors in that cache.
On the second pass, we are able to confirm that the select from table T4 is not yet in the session cursor cache. Immediately after the two cursors selected from table T2 were opened, we see that the session experienced one more cursor cache hit, and another cursor was added to the session cursor cache.
At the start of the third pass, we see that our select from T2 is now in the session cursor cache, not once, but twice, there were 50 cursors in the session cursor cache, and there was one more session cursor cache hit. When the two cursors selecting from table T4 were opened, we see that the session experienced three more cursor cache hits.
Let’s take a quick look at portions of the 10046 trace file that are related to the SQL statements that were executed by our script (just primarily focusing on the parse and close calls):
PARSING IN CURSOR #429272512 len=212 dep=0 uid=64 oct=3 lid=64 tim=2769436992465 hv=3255970530 ad='3ed509198' sqlid='c2a4dv3114ar2'
SELECT
SN.NAME STAT_NAME,
M.VALUE
FROM
V$MYSTAT M,
V$STATNAME SN
WHERE
SN.NAME IN ('session cursor cache hits','session cursor cache count')
AND SN.STATISTIC#=M.STATISTIC#
ORDER BY
SN.NAME
END OF STMT
CLOSE #429272512:c=0,e=16,dep=0,type=0,tim=2769436996834
PARSING IN CURSOR #429272512 len=161 dep=0 uid=64 oct=3 lid=64 tim=2769437006931 hv=3838443513 ad='3eb511630' sqlid='babcumbkcmzzt'
SELECT
SQL_TEXT
FROM
V$OPEN_CURSOR
WHERE
SID=
(SELECT
SID
FROM
V$MYSTAT
WHERE
ROWNUM=1)
ORDER BY
SQL_TEXT
END OF STMT
CLOSE #429272512:c=0,e=19,dep=0,type=0,tim=2769437013363
PARSING IN CURSOR #429272512 len=16 dep=0 uid=64 oct=3 lid=64 tim=2769437025761 hv=1886046789 ad='3eba6b810' sqlid='49fgb3ts6pkk5'
SELECT * FROM T4
END OF STMT
PARSING IN CURSOR #441175000 len=16 dep=0 uid=64 oct=3 lid=64 tim=2769437031138 hv=1886046789 ad='3eba6b810' sqlid='49fgb3ts6pkk5'
SELECT * FROM T4
END OF STMT
PARSING IN CURSOR #440835720 len=212 dep=0 uid=64 oct=3 lid=64 tim=2769437036683 hv=3255970530 ad='3ed509198' sqlid='c2a4dv3114ar2'
SELECT
SN.NAME STAT_NAME,
M.VALUE
FROM
V$MYSTAT M,
V$STATNAME SN
WHERE
SN.NAME IN ('session cursor cache hits','session cursor cache count')
AND SN.STATISTIC#=M.STATISTIC#
ORDER BY
SN.NAME
END OF STMT
CLOSE #441175000:c=0,e=11,dep=0,type=0,tim=2769437040272
CLOSE #429272512:c=0,e=8,dep=0,type=0,tim=2769437040299
CLOSE #440835720:c=0,e=6,dep=0,type=0,tim=2769437040318
=====================
PARSING IN CURSOR #429272512 len=212 dep=0 uid=64 oct=3 lid=64 tim=2769437040366 hv=3255970530 ad='3ed509198' sqlid='c2a4dv3114ar2'
SELECT
SN.NAME STAT_NAME,
M.VALUE
FROM
V$MYSTAT M,
V$STATNAME SN
WHERE
SN.NAME IN ('session cursor cache hits','session cursor cache count')
AND SN.STATISTIC#=M.STATISTIC#
ORDER BY
SN.NAME
END OF STMT
CLOSE #429272512:c=0,e=13,dep=0,type=1,tim=2769437042471
=====================
PARSING IN CURSOR #441175000 len=161 dep=0 uid=64 oct=3 lid=64 tim=2769437042547 hv=3838443513 ad='3eb511630' sqlid='babcumbkcmzzt'
SELECT
SQL_TEXT
FROM
V$OPEN_CURSOR
WHERE
SID=
(SELECT
SID
FROM
V$MYSTAT
WHERE
ROWNUM=1)
ORDER BY
SQL_TEXT
END OF STMT
CLOSE #441175000:c=0,e=19,dep=0,type=0,tim=2769437051038
=====================
PARSING IN CURSOR #441175000 len=16 dep=0 uid=64 oct=3 lid=64 tim=2769437051137 hv=1886046789 ad='3eba6b810' sqlid='49fgb3ts6pkk5'
SELECT * FROM T4
END OF STMT
PARSING IN CURSOR #440600648 len=16 dep=0 uid=64 oct=3 lid=64 tim=2769437053863 hv=1886046789 ad='3eba6b810' sqlid='49fgb3ts6pkk5'
SELECT * FROM T4
END OF STMT
CLOSE #440600648:c=0,e=10,dep=0,type=1,tim=2769437060397
CLOSE #441175000:c=0,e=14,dep=0,type=1,tim=2769437060431
CLOSE #429272512:c=0,e=14,dep=0,type=3,tim=2769437060459
PARSE #429272512:c=0,e=15,p=0,cr=0,cu=0,mis=0,r=0,dep=0,og=1,plh=1338012530,tim=2769437060499
CLOSE #429272512:c=0,e=31,dep=0,type=3,tim=2769437062969
=====================
PARSING IN CURSOR #440639688 len=161 dep=0 uid=64 oct=3 lid=64 tim=2769437063067 hv=3838443513 ad='3eb511630' sqlid='babcumbkcmzzt'
SELECT
SQL_TEXT
FROM
V$OPEN_CURSOR
WHERE
SID=
(SELECT
SID
FROM
V$MYSTAT
WHERE
ROWNUM=1)
ORDER BY
SQL_TEXT
END OF STMT
CLOSE #440639688:c=0,e=20,dep=0,type=1,tim=2769437071488
PARSE #440600648:c=0,e=25,p=0,cr=0,cu=0,mis=0,r=0,dep=0,og=1,plh=2560505625,tim=2769437071547
PARSE #441175000:c=0,e=16,p=0,cr=0,cu=0,mis=0,r=0,dep=0,og=1,plh=2560505625,tim=2769437073533
PARSE #429272512:c=0,e=19,p=0,cr=0,cu=0,mis=0,r=0,dep=0,og=1,plh=1338012530,tim=2769437076014
CLOSE #441175000:c=0,e=7,dep=0,type=3,tim=2769437079257
CLOSE #440600648:c=0,e=4,dep=0,type=3,tim=2769437079281
What does the above show? Notice that the numbers following type= on the CLOSE lines seem to change – what do those numbers mean? If we refer to page 284 of the book “Secrets of the Oracle Database“, we find the following information:
type=0: cursor is actually closed.
type=1: cursor is added to the session cursor cache without pushing an existing cursor out of the session cursor cache.
type=2: cursor is added to the session cursor cache, in the process pushing an existing cursor out of the session cursor cache.
type=3: cursor was already in the session cursor cache
So, the above indicates that the cursors are actually added to the session cursor cache NOT during a parse call, but instead when the cursor is attempted to be explicitly closed. The documentation describing the SESSION_CACHED_CURSORS parameter, referenced above, appears to be misleading. It is a bit odd that our select from T4 was added to the session cursor cache twice. It is also a bit odd that the large cursor numbers, introduced with Oracle Database 11.2.0.2 that I understand to represent the address of the cursor, were reused for different SQL statements (this might have significance in a later blog article).
Just for fun, let’s re-execute the VBS script:
CSCRIPT SessionCachedCursorsTest.vbs
This is now the script’s output:
SID: 192, SERIAL#: 693
Before Pass #1
session cursor cache count 2
session cursor cache hits 1
1 SELECT SN.NAME STAT_NAME, M.VALUE FROM V$MYSTAT M,
2 SELECT SQL_TEXT FROM V$OPEN_CURSOR WHERE SID=
3 insert into sys.aud$( sessionid,entryid,statement,ntimestamp
4 select /*+ connect_by_filtering */ privilege#,level from sys
5 select privilege# from sysauth$ where (grantee#=:1 or grante
After Open Pass #1
session cursor cache count 4
session cursor cache hits 2
Before Pass #2
session cursor cache count 6
session cursor cache hits 3
1 SELECT SN.NAME STAT_NAME, M.VALUE FROM V$MYSTAT M,
2 SELECT SQL_TEXT FROM V$OPEN_CURSOR WHERE SID=
3 SELECT * FROM T4
4 SELECT * FROM T4
5 insert into sys.aud$( sessionid,entryid,statement,ntimestamp
6 select /*+ connect_by_filtering */ privilege#,level from sys
7 select privilege# from sysauth$ where (grantee#=:1 or grante
After Open Pass #2
session cursor cache count 6
session cursor cache hits 7
Before Pass #3
session cursor cache count 6
session cursor cache hits 8
1 SELECT SN.NAME STAT_NAME, M.VALUE FROM V$MYSTAT M,
2 SELECT SQL_TEXT FROM V$OPEN_CURSOR WHERE SID=
3 SELECT * FROM T4
4 SELECT * FROM T4
5 insert into sys.aud$( sessionid,entryid,statement,ntimestamp
6 select /*+ connect_by_filtering */ privilege#,level from sys
7 select privilege# from sysauth$ where (grantee#=:1 or grante
After Open Pass #3
session cursor cache count 6
session cursor cache hits 12
In the above, you will likely first notice that the SID and SERIAL# for our newly created session is not the same as it was earlier. At the start of the output, before the first execution of the select from T4, we see that there were two cursors in the session’s cursor cache, and one hit on one of those two cursors – quite a change from the first execution of the script. Immediately after opening cursors that selected from T4 we see that there were 4 cursors in the session cursor count, and 2 hits on those cursors.
Now another big change before the start of the second pass, we see that there were 6 cursors in the session cursor cache, including two selects from T4 – previously, those cursors were added to the session cursor cache just before the start of the third pass. After opening the two cursors selecting from T4, there were 4 more hits on the session cursor cache.
So, what are the apparent take-aways from this test case:
- After flushing the shared pool a session’s cursor cache may contain a large number of SQL statements, duplicated in some cases, that may have only been executed a single time by the session at a depth greater than 0 (as indicated in a 10046 trace file).
- After a SQL statement is parsed and closed 3 times the SQL statement is eligible for the session cursor cache when the SQL statement is closed for the third time.
- If a SQL statement was previously parsed and closed 3 times by another session, the SQL statement is eligible for a different session’s cursor cache the first time that a cursor with that SQL statement is closed in the session.
- Oracle Database 11.2.0.2’s huge cursor numbers that are written to 10046 trace files may be reused for entirely different SQL statements, and that does not necessarily indicate a cursor leak.
- If a SQL statement is opened multiple times in a session without being closed between opens, it is quite possible that all of the cursors using the SQL statement will be added to the session cached cursors (the SQL statement will appear multiple times in V$OPEN_CURSOR) when those cursors are explicitly closed.
—
At this point, you are probably wondering what might happen when we open 12 cursors that select from T4. Here is the modified script (script download, save with .vbs extension SessionCachedCursorsTest2.vbs):
Dim strSQL
Dim strSQL2
Dim strSQL3
Dim strUsername
Dim strPassword
Dim strDatabase
Dim snpData
Dim snpData2
Dim snpData3
Dim snpData4
Dim snpData5
Dim snpData6
Dim snpData7
Dim snpData8
Dim snpData9
Dim snpData10
Dim snpData11
Dim snpData12
Dim snpDataCursorCache
Dim snpDataOpenSQL
Dim snpDataSID
Dim dbDatabase
Dim lngCnt
Dim intPass
Dim objFSO
Dim objFile
Set objFSO = CreateObject("Scripting.FileSystemObject")
Set objFile=objFSO.CreateTextFile("C:\SessionCursorCache2.txt", True)
Set snpData = CreateObject("ADODB.Recordset")
Set snpData2 = CreateObject("ADODB.Recordset")
Set snpData3 = CreateObject("ADODB.Recordset")
Set snpData4 = CreateObject("ADODB.Recordset")
Set snpData5 = CreateObject("ADODB.Recordset")
Set snpData6 = CreateObject("ADODB.Recordset")
Set snpData7 = CreateObject("ADODB.Recordset")
Set snpData8 = CreateObject("ADODB.Recordset")
Set snpData9 = CreateObject("ADODB.Recordset")
Set snpData10 = CreateObject("ADODB.Recordset")
Set snpData11 = CreateObject("ADODB.Recordset")
Set snpData12 = CreateObject("ADODB.Recordset")
Set snpDataCursorCache = CreateObject("ADODB.Recordset")
Set snpDataOpenSQL = CreateObject("ADODB.Recordset")
Set snpDataSID = CreateObject("ADODB.Recordset")
Set dbDatabase = CreateObject("ADODB.Connection")
strUsername = "MyUsername"
strPassword = "MyPassword"
strDatabase = "MyDatabase"
dbDatabase.ConnectionString = "Provider=OraOLEDB.Oracle;Data Source=" & strDatabase & ";User ID=" & strUsername & ";Password=" & strPassword & ";FetchSize=5000;"
dbDatabase.Open
strSQL = "SELECT" & VBCrLf
strSQL = strSQL & " S.SID," & VBCrLf
strSQL = strSQL & " S.SERIAL# SERIALN" & VBCrLf
strSQL = strSQL & "FROM" & VBCrLf
strSQL = strSQL & " V$SESSION S," & VBCrLf
strSQL = strSQL & " (SELECT" & VBCrLf
strSQL = strSQL & " SID" & VBCrLf
strSQL = strSQL & " FROM" & VBCrLf
strSQL = strSQL & " V$MYSTAT" & VBCrLf
strSQL = strSQL & " WHERE" & VBCrLf
strSQL = strSQL & " ROWNUM=1) M" & VBCrLf
strSQL = strSQL & "WHERE" & VBCrLf
strSQL = strSQL & " S.SID=M.SID" & VBCrLf
snpDataSID.Open strSQL, dbDatabase
If snpDataSID.State = 1 Then
Do While Not(snpDataSID.EOF)
objFile.Write "SID: " & snpDataSID("sid") & ", SERIAL#: " & snpDataSID("serialn") & VBCrLf
snpDataSID.MoveNext
Loop
snpDataSID.Close
objFile.Write " " & VBCrLf
End If
dbDatabase.Execute "ALTER SESSION SET TRACEFILE_IDENTIFIER = 'SessionCursorCache2'"
dbDatabase.Execute "ALTER SESSION SET EVENTS '10046 TRACE NAME CONTEXT FOREVER, LEVEL 1'"
strSQL2 = "SELECT" & VBCrLf
strSQL2 = strSQL2 & " SN.NAME STAT_NAME," & VBCrLf
strSQL2 = strSQL2 & " M.VALUE" & VBCrLf
strSQL2 = strSQL2 & "FROM" & VBCrLf
strSQL2 = strSQL2 & " V$MYSTAT M," & VBCrLf
strSQL2 = strSQL2 & " V$STATNAME SN" & VBCrLf
strSQL2 = strSQL2 & "WHERE" & VBCrLf
strSQL2 = strSQL2 & " SN.NAME IN ('session cursor cache hits','session cursor cache count')" & VBCrLf
strSQL2 = strSQL2 & " AND SN.STATISTIC#=M.STATISTIC#" & VBCrLf
strSQL2 = strSQL2 & "ORDER BY" & VBCrLf
strSQL2 = strSQL2 & " SN.NAME"
strSQL3 = "SELECT" & VBCrLf
strSQL3 = strSQL3 & " SQL_TEXT" & VBCrLf
strSQL3 = strSQL3 & "FROM" & VBCrLf
strSQL3 = strSQL3 & " V$OPEN_CURSOR" & VBCrLf
strSQL3 = strSQL3 & "WHERE" & VBCrLf
strSQL3 = strSQL3 & " SID=" & VBCrLf
strSQL3 = strSQL3 & " (SELECT" & VBCrLf
strSQL3 = strSQL3 & " SID" & VBCrLf
strSQL3 = strSQL3 & " FROM" & VBCrLf
strSQL3 = strSQL3 & " V$MYSTAT" & VBCrLf
strSQL3 = strSQL3 & " WHERE" & VBCrLf
strSQL3 = strSQL3 & " ROWNUM=1)" & VBCrLf
strSQL3 = strSQL3 & "ORDER BY" & VBCrLf
strSQL3 = strSQL3 & " SQL_TEXT"
For intPass = 1 to 3
snpDataCursorCache.Open strSQL2, dbDatabase
If snpDataCursorCache.State = 1 Then
objFile.Write "Before Pass #" & intPass & VBCrLf
Do While Not(snpDataCursorCache.EOF)
objFile.Write snpDataCursorCache("stat_name") & " " & snpDataCursorCache("value") & VBCrLf
snpDataCursorCache.MoveNext
Loop
snpDataCursorCache.Close
objFile.Write " " & VBCrLf
End If
snpDataOpenSQL.Open strSQL3, dbDatabase
lngCnt = 0
If snpDataOpenSQL.State = 1 Then
Do While Not(snpDataOpenSQL.EOF)
lngCnt = lngCnt + 1
objFile.Write lngCnt & " " & snpDataOpenSQL("sql_text") & VBCrLf
snpDataOpenSQL.MoveNext
Loop
snpDataOpenSQL.Close
objFile.Write " " & VBCrLf
End If
strSQL = "SELECT * FROM T4"
snpData.Open strSQL, dbDatabase
snpData2.Open strSQL, dbDatabase
snpData3.Open strSQL, dbDatabase
snpData4.Open strSQL, dbDatabase
snpData5.Open strSQL, dbDatabase
snpData6.Open strSQL, dbDatabase
snpData7.Open strSQL, dbDatabase
snpData8.Open strSQL, dbDatabase
snpData9.Open strSQL, dbDatabase
snpData10.Open strSQL, dbDatabase
snpData11.Open strSQL, dbDatabase
snpData12.Open strSQL, dbDatabase
snpDataCursorCache.Open strSQL2, dbDatabase
If snpDataCursorCache.State = 1 Then
objFile.Write "After Open Pass #" & intPass & VBCrLf
Do While Not(snpDataCursorCache.EOF)
objFile.Write snpDataCursorCache("stat_name") & " " & snpDataCursorCache("value") & VBCrLf
snpDataCursorCache.MoveNext
Loop
snpDataCursorCache.Close
objFile.Write " " & VBCrLf
End If
lngCnt = 0
If snpData.State = 1 Then
Do While Not(snpData.EOF)
Do While (lngCnt < 1000) and Not(snpData.EOF)
lngCnt = lngCnt + 1
snpData.MoveNext
Loop
lngCnt = 0
Do While (lngCnt < 1000) and Not(snpData2.EOF)
lngCnt = lngCnt + 1
snpData2.MoveNext
Loop
lngCnt = 0
Do While (lngCnt < 1000) and Not(snpData3.EOF)
lngCnt = lngCnt + 1
snpData3.MoveNext
Loop
lngCnt = 0
Do While (lngCnt < 1000) and Not(snpData4.EOF)
lngCnt = lngCnt + 1
snpData4.MoveNext
Loop
lngCnt = 0
Do While (lngCnt < 1000) and Not(snpData5.EOF)
lngCnt = lngCnt + 1
snpData5.MoveNext
Loop
lngCnt = 0
Do While (lngCnt < 1000) and Not(snpData6.EOF)
lngCnt = lngCnt + 1
snpData6.MoveNext
Loop
lngCnt = 0
Do While (lngCnt < 1000) and Not(snpData7.EOF)
lngCnt = lngCnt + 1
snpData7.MoveNext
Loop
lngCnt = 0
Do While (lngCnt < 1000) and Not(snpData8.EOF)
lngCnt = lngCnt + 1
snpData8.MoveNext
Loop
lngCnt = 0
Do While (lngCnt < 1000) and Not(snpData9.EOF)
lngCnt = lngCnt + 1
snpData9.MoveNext
Loop
lngCnt = 0
Do While (lngCnt < 1000) and Not(snpData10.EOF)
lngCnt = lngCnt + 1
snpData10.MoveNext
Loop
lngCnt = 0
Do While (lngCnt < 1000) and Not(snpData11.EOF)
lngCnt = lngCnt + 1
snpData11.MoveNext
Loop
lngCnt = 0
Do While (lngCnt < 1000) and Not(snpData12.EOF)
lngCnt = lngCnt + 1
snpData12.MoveNext
Loop
lngCnt = 0
Loop
End If
snpData.Close
snpData2.Close
snpData3.Close
snpData4.Close
snpData5.Close
snpData6.Close
snpData7.Close
snpData8.Close
snpData9.Close
snpData10.Close
snpData11.Close
snpData12.Close
Next
dbDatabase.Execute "ALTER SESSION SET EVENTS '10046 TRACE NAME CONTEXT OFF'"
objFile.Close
Set snpData = Nothing
Set snpData2 = Nothing
Set snpData3 = Nothing
Set snpData4 = Nothing
Set snpData5 = Nothing
Set snpData6 = Nothing
Set snpData7 = Nothing
Set snpData8 = Nothing
Set snpData9 = Nothing
Set snpData10 = Nothing
Set snpData11 = Nothing
Set snpData12 = Nothing
Set snpDataCursorCache = Nothing
Set snpDataOpenSQL = Nothing
Set snpDataSID = Nothing
dbDatabase.Close
Set dbDatabase = Nothing
Set objFile = Nothing
Set objFSO = Nothing
The following shows the output that I received on the first execution (the shared pool was not flushed prior to executing the script):
SID: 159, SERIAL#: 5853
Before Pass #1
session cursor cache count 7
session cursor cache hits 1
1 SELECT S.SID, S.SERIAL# SERIALN FROM V$SESSION S,
2 SELECT SN.NAME STAT_NAME, M.VALUE FROM V$MYSTAT M,
3 SELECT SQL_TEXT FROM V$OPEN_CURSOR WHERE SID=
4 SELECT PARAMETER, VALUE FROM SYS.NLS_DATABASE_PARAMETERS WHE
5 insert into sys.aud$( sessionid,entryid,statement,ntimestamp
6 select /*+ connect_by_filtering */ privilege#,level from sys
7 select SYS_CONTEXT('USERENV', 'SERVER_HOST'), SYS_CONTEXT('U
8 select decode(failover_method, NULL, 0 , 'BASIC', 1, 'PRECON
9 select privilege# from sysauth$ where (grantee#=:1 or grante
10 select value$ from props$ where name = 'GLOBAL_DB_NAME'
After Open Pass #1
session cursor cache count 9
session cursor cache hits 2
Before Pass #2
session cursor cache count 21
session cursor cache hits 3
1 SELECT S.SID, S.SERIAL# SERIALN FROM V$SESSION S,
2 SELECT SN.NAME STAT_NAME, M.VALUE FROM V$MYSTAT M,
3 SELECT SQL_TEXT FROM V$OPEN_CURSOR WHERE SID=
4 SELECT * FROM T4
5 SELECT * FROM T4
6 SELECT * FROM T4
7 SELECT * FROM T4
8 SELECT * FROM T4
9 SELECT * FROM T4
10 SELECT * FROM T4
11 SELECT * FROM T4
12 SELECT * FROM T4
13 SELECT * FROM T4
14 SELECT * FROM T4
15 SELECT * FROM T4
16 SELECT PARAMETER, VALUE FROM SYS.NLS_DATABASE_PARAMETERS WHE
17 insert into sys.aud$( sessionid,entryid,statement,ntimestamp
18 select /*+ connect_by_filtering */ privilege#,level from sys
19 select SYS_CONTEXT('USERENV', 'SERVER_HOST'), SYS_CONTEXT('U
20 select decode(failover_method, NULL, 0 , 'BASIC', 1, 'PRECON
21 select privilege# from sysauth$ where (grantee#=:1 or grante
22 select value$ from props$ where name = 'GLOBAL_DB_NAME'
After Open Pass #2
session cursor cache count 21
session cursor cache hits 17
Before Pass #3
session cursor cache count 21
session cursor cache hits 18
1 SELECT S.SID, S.SERIAL# SERIALN FROM V$SESSION S,
2 SELECT SN.NAME STAT_NAME, M.VALUE FROM V$MYSTAT M,
3 SELECT SQL_TEXT FROM V$OPEN_CURSOR WHERE SID=
4 SELECT * FROM T4
5 SELECT * FROM T4
6 SELECT * FROM T4
7 SELECT * FROM T4
8 SELECT * FROM T4
9 SELECT * FROM T4
10 SELECT * FROM T4
11 SELECT * FROM T4
12 SELECT * FROM T4
13 SELECT * FROM T4
14 SELECT * FROM T4
15 SELECT * FROM T4
16 SELECT PARAMETER, VALUE FROM SYS.NLS_DATABASE_PARAMETERS WHE
17 insert into sys.aud$( sessionid,entryid,statement,ntimestamp
18 select /*+ connect_by_filtering */ privilege#,level from sys
19 select SYS_CONTEXT('USERENV', 'SERVER_HOST'), SYS_CONTEXT('U
20 select decode(failover_method, NULL, 0 , 'BASIC', 1, 'PRECON
21 select privilege# from sysauth$ where (grantee#=:1 or grante
22 select value$ from props$ where name = 'GLOBAL_DB_NAME'
After Open Pass #3
session cursor cache count 21
session cursor cache hits 32
Is that what you expected to see?
—
We still have one remaining question – is a larger shared pool required when increasing the SESSION_CACHED_CURSORS parameter (the parameter was set to the default value of 50 for this blog article). To help you decide, refer to the following:
AskTom Article
Blog article written by Jonathan Lewis, specifically this comment:
“Every individiual holding a cursor open has an entry in x$kgllk – which is in the SGA – and these entries seem to be 172 bytes long in 10g (152 in 9i). So, clearly, if you hold more cursors open, you will be using more memory for these structures.”
Recent Comments