Oracle 数据库执行计划的查看与分析技巧

目录

  • Oracle 数据库执行计划的查看与分析技巧
    • 一、什么是执行计划
    • 二、查看执行计划的方法
      • (一)使用 EXPLAIN PLAN 命令
      • (二)通过 SQL Developer 工具查看
      • (三)启用 AUTOTRACE 功能
    • 三、执行计划中的关键信息解读
      • (一)操作类型
        • 全表扫描(TABLE ACCESS FULL)
        • 索引扫描(INDEX SCAN)
        • 嵌套循环连接(NESTED LOOPS)
        • 哈希连接(HASH JOIN)
      • (二)执行顺序
      • (三)谓词信息
    • 四、分析执行计划的技巧
      • (一)关注高成本操作
      • (二)结合数据量与分布情况
      • (三)对比不同执行计划版本
    • 五、优化执行计划的案例
    • 总结

Oracle 数据库执行计划的查看与分析技巧

在 Oracle 数据库中,执行计划能够帮助我们深入了解 SQL 语句在数据库内部的执行细节,进而优化查询性能、提升系统效率。无论是数据库领域的新手,还是经验丰富的工程师,掌握执行计划的查看与分析方法都至关重要。

一、什么是执行计划

执行计划是 Oracle 数据库优化器为 SQL 语句生成的一种执行蓝图,它描述了数据库将如何检索数据以满足查询要求。简单来说,执行计划告诉我们 SQL 语句的各个步骤,例如通过哪些索引进行数据查找、表之间以何种连接方式关联、数据如何排序等操作的先后顺序。优化器会基于数据库对象的统计信息、SQL 语句的语法结构以及数据库的配置参数等因素,综合考量来生成它认为最优的执行计划。

二、查看执行计划的方法

(一)使用 EXPLAIN PLAN 命令

这是最基础、也是最常用的查看执行计划的方式之一。它的语法如下:

EXPLAIN PLAN FOR
<your_sql_statement>;SELECT * FROM TABLE(DBMS_XPLAN.DISPLAY);

例如,我们有一个简单的查询语句,用于从员工表(employees)和部门表(departments)中检索特定部门的员工信息:

EXPLAIN PLAN FOR
SELECT e.employee_id, e.first_name, e.last_name, d.department_name
FROM employees e
JOIN departments d ON e.department_id = d.department_id
WHERE d.department_name = 'Sales';SELECT * FROM TABLE(DBMS_XPLAN.DISPLAY);

执行上述代码后,第二句查询会以表格形式展示出详细的执行计划。其中包括各操作的 ID、操作名称(如 TABLE ACCESS FULL 表示全表扫描,INDEX RANGE SCAN 表示索引范围扫描等)、对象名称(涉及的表或索引)以及执行顺序等关键信息。

(二)通过 SQL Developer 工具查看

SQL Developer 是 Oracle 官方提供的一款功能强大的数据库开发工具。在使用它执行 SQL 语句时,可以方便地同时查看对应的执行计划。只需在执行 SQL 的窗口中,点击 “解释计划” 按钮(通常是一个带有放大镜和闪电标志的图标),工具就会在下方的面板中以可视化的树状结构展示执行计划。这种方式相较于命令行,更加直观,易于理解。各个节点展示了详细的操作信息,并且可以通过鼠标悬停查看更多细节,如谓词信息(WHERE 子句中的过滤条件)等。

(三)启用 AUTOTRACE 功能

在 SQL*Plus 环境下,我们可以启用 AUTOTRACE 来查看执行计划及相关的执行统计信息,如物理读、逻辑读、执行时间等。首先需要确保当前用户具有执行 AUTOTRACE 相关权限,并且数据库实例已正确配置。启用 AUTOTRACE 的命令如下:

SET AUTOTRACE ON;

之后执行 SQL 语句,例如:

SELECT * FROM customers WHERE customer_city = 'New York';

执行完 SQL 后,除了返回查询结果,还会输出执行计划的概要信息以及上述提到的统计信息。这对于快速评估 SQL 语句的性能开销非常有帮助。要关闭 AUTOTRACE 功能,使用:

SET AUTOTRACE OFF;

三、执行计划中的关键信息解读

(一)操作类型

全表扫描(TABLE ACCESS FULL)

这意味着数据库会读取表中的所有行来满足查询条件。当没有合适的索引可用,或者优化器认为全表扫描的成本更低时,会选择这种方式。例如,在一个数据量较小的表上进行没有过滤条件或过滤条件选择性很差的,全表扫描可能是最快的方法。但对于大表,全表扫描通常会导致大量的 I/O 操作,严重影响性能。

索引扫描(INDEX SCAN)

又分为索引唯一扫描(INDEX UNIQUE SCAN)、索引范围扫描(INDEX RANGE SCAN)等。索引唯一扫描用于查找具有唯一键值的行,比如通过主键查询单条记录。索引范围扫描则适用于基于某个范围条件的查询,如查询某个时间段内的数据,它会利用索引的有序性快速定位到符合条件的起始和结束位置,并扫描其间的索引条目。

嵌套循环连接(NESTED LOOPS)

这是一种常见的表连接方式,对于外部表的每一行,都会在内层表中查找匹配的行。它适用于连接条件选择性高、关联表数据量较小的场景。优点是能快速返回少量精确匹配的结果,但如果表数据量大,可能会产生大量的循环操作,性能急剧下降。

哈希连接(HASH JOIN)

先对一张表构建哈希表,然后利用哈希函数快速查找另一张表中匹配的行。通常在连接大数据集时表现较好,尤其是当两张表都比较大且没有合适索引的情况下,哈希连接能通过减少数据比较次数来提高连接效率。

(二)执行顺序

执行计划中的操作 ID 标识了各操作的执行顺序,通常是从缩进少的节点开始,逐步向缩进多的节点推进。数字越小,执行优先级越高。通过观察执行顺序,我们可以了解数据的流动方向,以及哪些操作是基础,哪些是后续基于前面结果的进一步处理。例如,先进行表的访问操作获取原始数据,然后可能进行过滤、连接等操作,最后进行排序或聚合等满足最终查询需求的步骤。

(三)谓词信息

谓词即 WHERE 子句中的过滤条件,在执行计划中会显示哪些谓词用于索引查找,哪些用于最终结果的过滤。如果某个谓词能够有效利用索引,说明该过滤条件具有较好的效果,可以快速缩小数据检索范围。反之,如果谓词只能在全表扫描后进行过滤,那可能需要考虑优化过滤条件或添加合适索引。例如,“WHERE column_name> 100 AND column_name < 200” 这样的范围谓词,若在索引列上,可能触发索引范围扫描;而 “WHERE function (column_name) = some_value”(函数作用于列上的条件),一般情况下会导致索引失效,引发全表扫描。

四、分析执行计划的技巧

(一)关注高成本操作

执行计划中的每个操作都有对应的成本估算,通常以 COST 值表示,包括 CPU 成本和 I/O 成本。重点关注成本较高的操作,这些往往是性能瓶颈所在。比如,当发现一个全表扫描操作的成本占比很大,且表数据量庞大时,就需要思考是否可以通过创建合适索引、优化查询条件等方式来改变执行计划,降低成本。可以通过对比不同优化方案下执行计划的成本变化,来评估优化效果。

(二)结合数据量与分布情况

了解表的实际数据量大小以及数据在索引列上的分布状况,对于准确分析执行计划至关重要。例如,一个索引在理论上看起来很完美,但如果表中的大部分数据在索引列上具有相同的值(数据倾斜),那么索引的选择性就会大打折扣,优化器可能会错误地选择使用这个低效的索引,导致性能问题。此时,可能需要考虑收集更准确的统计信息,或调整查询语句以适应数据分布特点,如增加额外的过滤条件来减少数据倾斜的影响。

(三)对比不同执行计划版本

在对 SQL 语句进行优化调整过程中,如修改索引、调整查询结构、更新数据库统计信息等操作后,重新查看并对比执行计划的变化。观察优化措施是否达到预期效果,新的执行计划中是否消除了高成本操作,数据检索路径是否更加合理。通过这种迭代式的对比分析,逐步逼近最优的查询性能。

五、优化执行计划的案例

假设我们有一个电商订单数据库,包含订单表(orders)、订单明细表(order_items)和产品表(products)。经常执行的查询是获取某个时间段内特定产品类别的订单总金额。初始查询语句如下:

SELECT p.product_category, SUM(oi.quantity * oi.unit_price) AS total_amount
FROM orders o
JOIN order_items oi ON o.order_id = oi.order_id
JOIN products p ON oi.product_id = p.product_id
WHERE o.order_date BETWEEN '2024-01-01' AND '2023-01-31'
AND p.product_category = 'Electronics'
GROUP BY p.product_category;

使用 EXPLAIN PLAN 查看执行计划后,发现存在以下问题:
对订单表(orders)进行了全表扫描,因为 order_date 列没有合适索引,导致大量不必要的 I/O 操作,查询效率低下。
在连接操作中,由于表之间的连接条件选择性不是特别高,且没有充分利用索引,嵌套循环连接的成本较高。
优化方案:
在订单表的 order_date 列上创建索引:

CREATE INDEX idx_order_date ON orders(order_date);

分析产品表(products)上 product_category 列的数据分布,发现该列数据存在一定倾斜,部分类别数据量远大于其他类别。考虑收集更精确的统计信息:

BEGINDBMS_STATS.GATHER_TABLE_STATS(ownname => 'your_schema', tabname => 'products');
END;

重新执行查询并查看执行计划,发现订单表改为使用索引范围扫描,大大减少了数据读取量;连接操作也因为统计信息的更新,优化器选择了更合适的哈希连接方式,整体查询性能提升了数倍,执行时间从原来的几十秒缩短到几秒。

总结

Oracle 数据库执行计划的查看与分析是数据库优化工作中的核心技能。通过熟练掌握多种查看执行计划的方法,深入解读其中的关键信息,并运用有效的分析技巧,我们能够精准定位 SQL 语句的性能问题,采取针对性的优化措施。从创建合适索引、优化查询语句结构,到确保准确的统计信息,每一个环节都可能成为提升数据库性能的关键。持续实践与经验积累,将帮助我们在面对复杂的数据库环境时,游刃有余地优化查询性能,保障系统高效稳定运行。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/891098.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【最新】沃德协会管理系统源码+uniapp前端+环境教程

一.系统介绍 一款基于FastAdminThinkPHPUniapp开发的商协会系统&#xff0c;新一代数字化商协会运营管理系统&#xff0c;以“智慧化会员体系、智敏化内容运营、智能化活动构建”三大板块为基点&#xff0c;实施功能全场景覆盖&#xff0c;一站式解决商协会需求壁垒&#xff0…

向bash shell脚本传参

例子&#xff1a; ~ script % touch parameter.sh ~ script % chmod 755 parameter.sh ~ % vim parameter.shparameter.sh: #!/usr/bin/env bashecho the name of current script is $0echo the first parameter is $1echo the second parameter is $2echo all parameters: $…

玩转树莓派Pico(20): 迷你气象站6——软件整合改进

前言 上次写的代码虽然能达到预期效果&#xff0c;但还是有很多问题的&#xff1a; 主程序main.py包含太多的内容&#xff0c;比较凌乱&#xff0c;因此整体设计要重新调整&#xff1b;没有日志功能&#xff0c;脱离电脑运行时根本不知道情况。比如有时断电重启后就不发送数据…

【LeetCode: 83. 删除排序链表中的重复元素 + 链表】

&#x1f680; 算法题 &#x1f680; &#x1f332; 算法刷题专栏 | 面试必备算法 | 面试高频算法 &#x1f340; &#x1f332; 越难的东西,越要努力坚持&#xff0c;因为它具有很高的价值&#xff0c;算法就是这样✨ &#x1f332; 作者简介&#xff1a;硕风和炜&#xff0c;…

算法练习——模拟题

前言&#xff1a;模拟题的特点在于没有什么固定的技巧&#xff0c;完全考验自己的代码能力&#xff0c;因此有助于提升自己的代码水平。如果说一定有什么技巧的话&#xff0c;那就是有的模拟题能够通过找规律来简化算法。 一&#xff1a;替换所有问号 题目要求&#xff1a; 解…

Idea创建JDK17的maven项目失败

Idea创建JDK17的maven项目失败 Error occurred during initialization of VM Could not find agent library instrument on the library path, with error: Can’t find dependent libraries Possible solution: Check your maven runner VM options. Open Maven Runner setti…

VSCode设置Playwright教程

1.安装扩展 打开VS Code&#xff0c;在扩展—>搜索"Playwright Test for VSCode"&#xff0c;点击安装 按快捷键CommandShiftP&#xff0c;输入install playwright&#xff0c;点击安装Playwright 安装成功会有如下提示 2.调试脚本 打开tests/example.spec.ts文…

HarmonyOS Next“说书人”项目 单机版 实践案例

前段时间开发了一个软件&#xff0c;取名为“说书人”&#xff0c;后由于备案暂时没有通过&#xff0c;于是删除了联网功能&#xff0c;重新做了一个单机版&#xff0c;这里对于单机版的开发实践案例进行一个发出&#xff0c;希望能帮助到大家 文章最后给出了AtomGit仓库地址 p…

HTML新特性|01 音频视频

音频 1、Audio (音频) HTML5提供了播放音频文件的标准 2、control(控制器) control 属性供添加播放、暂停和音量控件 3、标签: <audio> 定义声音 <source> 规定多媒体资源,可以是多个<!DOCTYPE html> <html lang"en"> <head><…

像素的访问和算术运算

【欢迎关注编码小哥&#xff0c;学习更多实用的编程方法和技巧】 一、常用的访问像素的方法 1、使用at()方法 // 灰度图 cv::Mat grayImage; for (int y 0; y < grayImage.rows; y) {for (int x 0; x < grayImage.cols; x) {uchar pixel grayImage.at<uchar>…

【深度学习】卷积网络代码实战ResNet

ResNet (Residual Network) 是由微软研究院的何凯明等人在2015年提出的一种深度卷积神经网络结构。ResNet的设计目标是解决深层网络训练中的梯度消失和梯度爆炸问题&#xff0c;进一步提高网络的表现。下面是一个ResNet模型实现&#xff0c;使用PyTorch框架来展示如何实现基本的…

js的讲解

Proxy 是 ES6&#xff08;ECMAScript 2015&#xff09;中引入的一个新的内置对象&#xff0c;用于定义某些操作的自定义行为&#xff08;如属性查找、赋值、枚举、函数调用等&#xff09;。Proxy 可以被看作是一个拦截器&#xff0c;它拦截并自定义对象上的基本操作。通过创建一…

雷电「模拟器」v9 最新清爽去广

前言 雷电模拟器9是基于安卓9内核开发的全新版本模拟器 安装环境 [名称]&#xff1a;雷电「模拟器」 [大小]&#xff1a;579MB [版本]&#xff1a;9.1.34 [语言]&#xff1a;简体中文 [安装环境]&#xff1a;Windows 通过网盘分享的文件&#xff1a;雷电模拟器 链接:…

大模型 API 接入初探

文章目录 大模型 API 接入初探一、使用大模型 API 的前置步骤&#xff08;一&#xff09;注册账户与获取凭证&#xff08;二&#xff09;理解 API 文档 二、三个常用 API&#xff08;一&#xff09;列出模型&#xff08;二&#xff09;FIM 补全&#xff08;三&#xff09;对话补…

ACPI PM Timer

ACPI PM Timer 概述&#xff1a; ACPI PM Timer是一个非常简单的计时器&#xff0c;它以 3.579545 MHz 运行&#xff0c;在计数器溢出时生成系统控制中断&#xff08;SCI&#xff09;。它精度较低&#xff0c;建议使用其他定时器&#xff0c;如HPET或APIC定时器。 检测ACPI P…

力扣--LCR 188.买卖芯片的最佳时机

题目 数组 prices 记录了某芯片近期的交易价格&#xff0c;其中 prices[i] 表示的 i 天该芯片的价格。你只能选择 某一天 买入芯片&#xff0c;并选择在 未来的某一个不同的日子 卖出该芯片。请设计一个算法计算并返回你从这笔交易中能获取的最大利润。 如果你不能获取任何利…

工业相机基本参数

分辨率&#xff08;Resolution&#xff09; 定义&#xff1a;分辨率指的是相机图像的像素数&#xff0c;通常以 宽度 x 高度 的形式表示&#xff0c;如 1920x1080 或 2592x1944。作用&#xff1a;分辨率越高&#xff0c;相机可以捕捉到更多的细节。高分辨率相机适用于需要精确…

实时在线翻译谷歌插件

Real - time Translation插件的安装 1、下载插件并解压 2、打开谷歌浏览器&#xff0c;在地址栏输入 “chrome://extensions/” 进入扩展程序页面. 3、开启页面右上角的 “开发者模式”. 4、点击 “加载已解压的扩展程序” 按钮&#xff0c;选择之前解压的文件夹&#xff0c;点…

torch.sparse_csc_tensor

torch.sparse_csc_tensor 以CSC格式构建一个稀疏张量。CSC格式的稀疏张量乘法运算通常比 COO 格式的稀疏张量更快。 CSC格式&#xff08;Compressed Sparse Column Format&#xff09;是一种存储稀疏矩阵的常用格式&#xff0c;它通过三个数组来表示稀疏矩阵&#xff1a; 非零…

C++ 设计模式:原型模式(Prototype Pattern)

链接&#xff1a;C 设计模式 链接&#xff1a;C 设计模式 - 工厂方法 链接&#xff1a;C 设计模式 - 抽象工厂 链接&#xff1a;C 设计模式 - 建造者模式 原型模式&#xff08;Prototype Pattern&#xff09;是一种创建型设计模式&#xff0c;它允许一个对象通过复制现有对象来…