详解六种方法查看oracle执行计划=》优化必备
mhr18 2024-09-26 14:01 17 浏览 0 评论
概述
很多时候我们去优化某条sql时,第一步通常是对sql做执行计划分析,看是走全扫还是索引扫描,是NL、SMJ还是HJ连接,是不是有绑定变量等等...今天不讲分析,主要讲怎么看执行计划。
下面介绍下几种常见的查看执行计划的方式(方法比较多,就不一一演示了)
1、 explain plan(类似plsql中的f5)
这里先使用explain plan命令对目标SQL做explain,在使用"select * from table(dbms_xplan.display)"查看上述使用explain plan命令后得到的执行计划。
PS:PL/SQL Developer中的快捷键F5就是在explain plan命令上的一层封装。
语法:explain plan for + SQL select * from table(dbms_xplan.display)
优点:无需真正执行,快捷方便
缺点:
a. 没有输出运行时的相关统计信息(产生多少逻辑读,多少次递归调用,多少次物理读的情况)
b. 无法判断是处理了多少行
c. 无法判断表被访问了多少次
其他命令
#配合explain plan使用 select * from table(dbms_xplan.display); #查看刚刚执行过的SQL的执行计划 select * from table(dbms_xplan.display_cursor(null,null,'advanced')); #只要目标SQL的执行计划所在的Child Cursor没有被age out出Shared Pool select sql_text,sql_id,hash_value,child_number from v$sql where sql_text like '%select * from emp%'; #用于查看指定SQL的执行计划 select * from table(dbms_xplan.display_cursor('sql_id/hash_value',child_cursor_number, 'advanced')); #VERSION_COUNT代表有几种不同的执行计划 select sql_text,sql_id,hash_value,child_number from v$sql where sql_text like '%select * from emp%'; #查看指定SQL的所有历史执行计划,前提是该SQL的执行计划被采集到AWR Repository中 select * from table(dbms_xplan.display_awr('sql_id'));
2、SQLPLUS中的AUTOTRACE
语法:set autotrace on
类似的有:
Set autotrace on #(得到执行计划,输出运行结果)
Set autotrace traceonly #(得到执行计划,不输出运行结果)
Set autotrace traceonly explain #(得到执行计划,不输出运行结果和统计信息部分,仅展现执行计划部分)
Set autotrace traceonly statistics #(不输出运行结果和执行计划部分,仅展现统计信息部分)
优点:
a. 可以输出运行时的相关统计信息(产生多少逻辑读,多少次递归调用,多少次物理读的情况)
b. 虽然必须要等语句执行完毕后才可输出执行计划,但是可以有traceonly开关来控制返回结果不打屏输出
缺点:
a. 必须等到语句真正执行完毕后,才可以出结果
b. 无法看到表被访问了多少次
3、statistics_level=all
语法:
alter session setstatistics_level=all; select * from table(dbms_xplan.display_cursor(null,null,'allstats last'));
注:
a. 如果用 /*+ gather_plan_statistics */的方法,可以省略alter session setstatistics_level=all;
b. 关键字解读:
Starts:该sql执行的次数。
E-Rows:执行计划预计的行数。
A-Rows:实际返回的行数。A-Rows跟E-Rows做比较,就可以确定哪一步执行计划出问题。
A-Time:每一步实际执行的时间(HH:MM:SS.FF),根据这一行可以知道该sql耗时在了哪个地方。
Buffers:每一步实际执行的逻辑读或一致性读。
Reads:物理读。
OMem:当前操作完成所有内存工作区(Work Aera)操作所总共使用私有内存(PGA)中工作区的大小,这个数据是由优化器统计数据以及前一次执行的性能数据估算得出的
1Mem:当工作区大小无法满足操作所需的大小时,需要将部分数据写入临时磁盘空间中(如果仅需要写入一次就可以完成操作,就称一次通过,One-Pass;否则为多次通过;Multi_Pass).该列数据为语句最后一次执行中,单次写磁盘所需要的内存大小,这个由优化器统计数据以及前一次执行的性能数据估算得出的
User-Mem:语句最后一次执行中,当前操作所使用的内存工作区大小,括号里面为(发生磁盘交换的次数,1次即为One-Pass,大于1次则为Multi_Pass,如果没有使用磁盘,则显示OPTIMAL)
OMem、1Mem为执行所需的内存评估值,0Mem为最优执行模式所需内存的评估值,1Mem为one-pass模式所需内存的评估值。0/1/M 为最优/one-pass/multipass执行的次数。Used-Mem耗的内存
优点:
a. 可以清晰的从starts得出表被访问多少
b. 可以清晰的从E-ROWS和A-ROWS中得到预测的行数和真实的行数,从而可以准确判断Oracle评估是否准确。
c. 虽然没有专门的输出运行时的相关统计信息,但是执行计划中的BUFFERS就是真实的逻辑读的多少
缺点:
a. 必须要等到语句真正执行完毕后,才可以出结果。
b. 无法控制记录输屏打出,不像autotrace有 traceonly 可以控制不将结果打屏输出。
c. 看不出递归调用的次数,看不出物理读的多少(不过逻辑读才是重点)
4、通过dbms_xplan.display_cursor输入sql_id参数直接获取
语法:
select * from table(dbms_xplan.display_cursor('&sq_id')); #从共享池获取
注:
a. select * fromtable(dbms_xplan.display_awr('&sq_id')); 从awr性能视图获取
b. 查看多个sql的执行计划
select * from table(dbms_xplan.display_cursor('4jj76r0vw14zx',0));
select * fromtable(dbms_xplan.display_cursor('4jj76r0vw14zx',1));
优点:
a. 知道sql_id立即可得到执行计划,和explain plan for 一样无需执行;
b. .可以得到真实的执行计划。
缺陷
a. 没有输出运行时的相关统计信息(产生多少逻辑读,多少次递归调用,多少次物理读的情况);
b. 无法判断是处理了多少行;
c. 无法判断表被访问了多少次。
5、10046事件
与之前三种查看执行计划方法不同之处在于,所得到的执行计划中明确显示了目标SQL实际执行计划中每一个执行步骤所消耗的逻辑读、物理读和花费的时间。执行计划与明细资源消耗会写入此Session所对应的trace文件中,Oracle会在参数USER_DUMP_DEST所代表的目录下生成这个trace文件。
- 首先在当前Session中激活10046事件
- 接着在此Session中执行目标SQL
- 最后在此Seesion中关闭10046事件
#激活10046事件两种方法 alter session set events '10046 trace name context forever,level 12' oradebug event 10046 trace name context forever,level 12 #关闭10046事件两种方法 alter session set events '10046 trace name conetxt off' oradebug event 10046 trace name context off #tkprof分析跟踪文件 tkprof trc文件 目标文件 sys=no sort=prsela,exeela,fchela (格式化命令)
不过我习惯用oradebug跟10046来做结合,下面简单演示下:
tkprof分析
优点:
a. 可以看出SQL语句对应的等待事件
b. 如果SQL语句中有函数调用,SQL中有SQL,将会都被列出,无处遁形。
c. 可以方便的看出处理的行数,产生的物理逻辑读。
d. 可以方便的看出解析时间和执行时间。
e. 可以跟踪整个程序包
缺陷:
a. 步骤繁琐,比较麻烦
b. 无法判断表被访问了多少次。
c. 执行计划中的谓词部分不能清晰的展现出来。
6. awrsqlrpt.sql
步骤:@?/rdbms/admin/awrsqrpt.sql
选择你要的断点(begin snap 和end snap)
输入sql_id
这样就可以得出结果了。
如果某SQL执行非常长时间才会出结果,甚至慢到返回不了结果,这时候看执行计划就只能用方法1;其实跟踪某条SQL最简单的方法是方法1,其次就是方法2;
如果想观察到某条SQL有多条执行计划的情况,只能用方法4和方法6;
如果SQL中含有多函数,函数中套有SQL等多层递归调用,想准确分析,只能使用方法5;
要想确保看到真实的执行计划,不能用方法1和方法2;
要想获取表被访问的次数,只能使用方法3;
后面会分享更多关于DBA方面内容,感兴趣的朋友可以关注下!
相关推荐
- 甲骨文签署多项大型云协议,其一未来可贡献超300亿美元年收入
-
IT之家7月1日消息,根据甲骨文Oracle当地时间6月30日向美国证券交易委员会(SEC)递交的FORM8-K文件,该企业在始于2025年6月1日的202...
- 甲骨文获TEMU巨额合同,后者大部分基础设施将迁移至Oracle云
-
IT之家6月23日消息,Oracle甲骨文创始人、董事长兼首席技术官LarryEllison(拉里埃里森)在本月早些时候的2025财年第四财季和全财年财报电话会议上表示,Oracle...
- Spring Boot 自定义数据源设置,这些坑你踩过吗?
-
你在使用SpringBoot进行后端开发的过程中,是不是也遇到过这样的问题:项目上线后,数据库连接总是不稳定,偶尔还会出现数据读取缓慢的情况,严重影响了用户体验。经过排查,发现很大一部分原因竟然...
- 一个开箱即用的代码生成器(一个开箱即用的代码生成器是什么)
-
今天给大家推荐一个好用的代码生成器,名为renren-generator,该项目附带前端页面,可以很方便的选择我们所需要生成代码的表。首先我们通过git工具克隆下来代码(地址见文末),导入idea。...
- 低代码建模平台-数据挖掘平台(低代码平台的实现方式)
-
现在来看一下数据连接。·这里是管理数据连接的空间,点击这里可以新增一个数据连接。·输入连接名称,然后输入url,是通过gdbc的方式去连接的数据库,目前是支持mysql、oracle以及国产数据库达梦...
- navicat 17.2.7连接oracle数据库提示加载oracle库失败
-
系统:macOS15.5navicat版本:navicatpremiumlite17.2.7连接oracle测试报错:加载oracle库失败【解决办法】:放达里面找到程序,显示简介里面勾选“使...
- 开源“Windows”ReactOS更新:支持全屏应用
-
IT之家6月17日消息,ReactOS团队昨日(6月16日)在X平台发布系列推文,公布了该系统的最新进展,包括升级Explorer组件,支持全屏应用,从Wine项目引入了...
- SSL 推出采用全模拟内置混音技术的模拟调音台Oracle
-
英国调音台传奇品牌SolidStateLogic宣布推出Oracle——一款采用全模拟内置混音技术的调音台,在紧凑的AWS尺寸机箱内集成了大型调音台的功能。该调音台提供24输入和...
- 47道网络工程师常见面试题,看看有没有你不会的!
-
你们好,我的网工朋友。网络工程师面试的时候,都会被问到什么?这个问题其实很泛,一般来说,你肯定要先看明白岗位需求写的是什么。基本上都是围绕公司需要的业务去问的。但不可否认的是,那些最基础的概念,多少也...
- 汉得信息:发布EBS系统安装启用JWS的高效解决方案
-
e公司讯,从汉得信息获悉,近日,微软官方宣布InternetExplorer桌面应用程序将于2022年6月15日正式停用。目前大部分客户都是使用IE浏览器打开EBS的Form界面,IE停用后,只能使...
- 36.9K star ! 推荐一个酷炫低代码开发平台!功能太强!
-
前言最近在逛github,看看能不能搜罗到一些对自己有帮助的开源软件。不经意间看到一个高star的java开源项目:jeecg-boot。进入在线演示版一看,感叹实在是太牛了!此开源项目不管是给来学习...
- Linux新手入门系列:Linux下jdk安装配置
-
本系列文章是把作者刚接触和学习Linux时候的实操记录分享出来,内容主要包括Linux入门的一些理论概念知识、Web程序、mysql数据库的简单安装部署,希望能够帮到一些初学者,少走一些弯路。注意:L...
- 手把手教你在嵌入式设备中使用SQLite3
-
摘要:数据库是用来存储和管理数据的专用软件,使得管理数据更加安全,方便和高效。数据库对数据的管理的基本单位是表(table),在嵌入式linux中有时候它也需要用到数据库,听起来好难,其实就是几个函数...
- JAVA语言基础(java语言基础知识)
-
一、计算机的基本概念什么是计算机?计算机(Computer)全称:电子计算机,俗称电脑。是一种能够按照程序运行、自动高速处理海量数据的现代化智能电子设备。由硬件和软件组成、没有安装过任何软件的计算机称...
- 再见 Navicat!一款开源的 Web 数据库管理工具!
-
大家好,我是Java陈序员。在日常的开发工作中,常常需要与各种数据库打交道。而为了提高工作效率,常常会使用一些可视化工具进行操作数据库。今天,给大家介绍一款开源的数据库管理工具,无需下载安装软件,基...
你 发表评论:
欢迎- 一周热门
- 最近发表
-
- 甲骨文签署多项大型云协议,其一未来可贡献超300亿美元年收入
- 甲骨文获TEMU巨额合同,后者大部分基础设施将迁移至Oracle云
- Spring Boot 自定义数据源设置,这些坑你踩过吗?
- 一个开箱即用的代码生成器(一个开箱即用的代码生成器是什么)
- 低代码建模平台-数据挖掘平台(低代码平台的实现方式)
- navicat 17.2.7连接oracle数据库提示加载oracle库失败
- 开源“Windows”ReactOS更新:支持全屏应用
- SSL 推出采用全模拟内置混音技术的模拟调音台Oracle
- 47道网络工程师常见面试题,看看有没有你不会的!
- 汉得信息:发布EBS系统安装启用JWS的高效解决方案
- 标签列表
-
- oracle位图索引 (74)
- oracle批量插入数据 (65)
- oracle事务隔离级别 (59)
- oracle 空为0 (51)
- oracle主从同步 (55)
- oracle 乐观锁 (51)
- redis 命令 (78)
- php redis (88)
- redis 存储 (66)
- redis 锁 (69)
- 启动 redis (66)
- redis 时间 (56)
- redis 删除 (67)
- redis内存 (57)
- redis并发 (52)
- redis 主从 (69)
- redis 订阅 (51)
- redis 登录 (54)
- redis 面试 (58)
- 阿里 redis (59)
- redis 搭建 (53)
- redis的缓存 (55)
- lua redis (58)
- redis 连接池 (61)
- redis 限流 (51)