细致入微:Oracle中执行计划在Shared Pool中的存储位置探秘

这两天我一直在想一个问题,那就是 Oracle 的执行计划到底存储在什么地儿?它会是一种什么样的格式?

这里我试图对这个问题做一点我自己认为的解释,这个解释可能是有问题的。

朋友们在看这篇文章之前,应该首先熟悉如下的这张图:

怎样才叫熟悉这张图呢?我认为验证的方法就是看你是否能够仅仅看着这张图,在1个小时的时间内把这张图的内容解释清楚。

这张图也许能够用来衡量你对 library cache 的了解程度。

我们现在做的测试其实就来源于上面这张图:

首先执行一下下述的 sql:

SQL_testdb>select * from scott.emp;

接着查一下上述 sql 在 library cache中的library cache object handle 的地址,一下两种方式都可以获取 SQL 语句父游标地址:

SQL_testdb>select address,sql_text from v$sqlarea where sql_text like ‘select * from scott.emp%’;

然后我们 dump 一下 library cache,注意这里level一定要大于等于8,否则看不到heap 0的内容:

SQL_testdb>alter session set events ‘immediate trace name library_cache level 11’; Session altered. SQL_testdb>oradebug setmypid Statement processed. SQL_testdb>oradebug tracefile_name /cadrasu01/app/oracle/admin/testdb/udump/testdb_ora_4128918.trc

从上述trace文件中我们以 070000001CAE2C70 为关键字去查询,查询结果如下:

注意看上述 library cache object 的类型是 cursor,名字是 select * from scott.emp,它有一个子 cursor,其 library cache object handle 的地址是 70000001cae15e0。我们现在就以这个地址继续搜索上述 trace 文件,搜到的内容如下:

从上述内容中我们可以看到,子 cursor 是没有名字的,这个其实很正常——因为 Oracle 是通过先访问其 parent cursor 后才会来访问它。

另外,这个子 cursor 只有两个 data block,分别是 data block 0 和 data block 6,对应的就是上图中的 heap 0 和 heap 6。

我首先排除掉 heap 0,理由如下:

The data block structure for a heap, stored in heap 0, contains a pointer to the first data block that is allocated for the heap, a status indicator, the pin under which the heap is loaded, and so on.

那么剩下的就只有一种可能,就是 Oracle 把 sql 的执行计划存储在了 heap 6里。

接下来我们 dump一下 heap 6 的内容:

SQL_testdb>oradebug setmypid Statement processed. SQL_testdb>alter session set events ‘immediate trace name heapdump_addr level 2, addr0x70000001cae1328′; Session altered. SQL_testdb>oradebug tracefile_name /cadrasu01/app/oracle/admin/testdb/udump/testdb_ora_2859106.trc

从上述 trace文件中我们可以看到如下的内容:

上述 trace 文件的内容告诉我们 heap 6 实际上就是 sql area(这个和上图中描述的 heap 6是 sql context 不尽相同),剩下的内容除了我们能看出包含了表 scott.emp 的各个字段的内容之外,其他的就看不懂了。

我猜测 Oracle 把 sql 的执行计划存在了这个 sql 的子 cursor 的 heap 6(也就是 sql area)中,只不过存储的形式是编译好的二进制格式。

感谢 MOS,让我找到了如下的论据,可以在某种程度上让我自圆其说:

Parsing a cursor builds four different library cache structures, if they do not already exist, within the library cache: 1、parent cursor handle 2、parent cursor object, containing the child dependency list 3、child cursor handle, inserted in the child dependency list of the parent object 4、child cursor object, containing the compilation and run-time execution plan for the compiled SQL statement.

原文发布于微信公众号 - 数据和云(OraNews)

原文发表时间:2016-07-14

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏阿杜的世界

JVM源码分析之perfData文件的创建

看泉子的一篇文章:JVM源码分析之Jstat工具原理完全解读 - 你假笨 里提到了两个JVM参数,可以控制perfdata文件是否共享,引用泉子对这两个参数的解...

771
来自专栏性能与架构

格式化SQL来提高效率

对SQL语句进行格式化不会让其运行得更快,但对我们的工作效率是有很大好处的 (1)提高SQL的可读性大大提高,便于维护 (2)提高开发效率,方便测试 例如 ? ...

3169
来自专栏维C果糖

史上最简单的 MySQL 教程(二十七)「连接查询(下)」

连接查询:将多张表(大于等于 2 张表)按照某个指定的条件进行数据的拼接,其最终结果记录数可能有变化,但字段数一定会增加。

3026
来自专栏FreeBuf

WIN10下ROP初体验

* 本文原创作者:与非门salome,本文属FreeBuf原创奖励计划,未经许可禁止转载 首先,在windows10下编写一个具有一定安全机制但又存在漏...

2049
来自专栏c#开发者

分析Oracle数据库日志文件(1)

分析Oracle数据库日志文件(1) 一、如何分析即LogMiner解释 从目前来看,分析Oracle日志的唯一方法就是使用Oracle公司提供的LogMin...

3295
来自专栏祝威廉

Effective PySpark(PySpark 常见问题)

首先确保安装了python 2.7 ,强烈建议你使用Virtualenv方便python环境的管理。之后通过pip 安装pyspark

663
来自专栏数据和云

深入解析:TRUNCATE TABLE 的内部原理解析与恢复思路

众所周知,truncate table 是一种快速清空表内数据的一种方式,与 delete 方式不同,truncate 只产生非常少的 redo 和 undo,...

501
来自专栏LEo的网络日志

c库函数getenv引起的core dumped

3498
来自专栏FreeBuf

Windows下SLmail邮件服务器缓冲区溢出理解及实验

本次缓冲区溢出实验是在Windows7 Unlimit 64位下的SLmail邮件服务溢出测试。 注:SLmail并不是一个特别常用的邮件服务应用,本次实验仅限...

2235
来自专栏杨建荣的学习笔记

关于date格式的两个案例(r4笔记第96天)

在工作中总是会碰到各种和date相关的问题,一般这种问题都是让人很纠结的。 比如前几天一个朋友和我分享了他关于时间问题的两个案例。 第一个是他在做impdp导入...

2944

扫描关注云+社区