MySQL 索引优化实战 – 结合 Explain 深度解析慢查询

在实际的开发过程中,随着数据量的不断增大,慢查询成为了不可忽视的性能瓶颈。MySQL 提供了多种工具来帮助我们分析查询性能问题,其中最常用的工具是 EXPLAINSHOW PROFILESHOW STATUS。本文将从慢查询日志入手,结合 EXPLAIN 解析慢查询,找出索引失效的原因,并提供相应的优化方案。通过本案例,您将掌握如何高效地优化 SQL 查询,提高 MySQL 的查询性能。


一、慢查询日志分析

慢查询日志是 MySQL 记录执行时间超过指定阈值的 SQL 语句的日志,主要用于分析性能瓶颈。要启用慢查询日志,可以通过以下命令设置:

-- 启用慢查询日志
SET GLOBAL slow_query_log = 'ON';-- 设置慢查询阈值为 2 秒
SET GLOBAL long_query_time = 2;

这样 MySQL 就会将执行超过 2 秒的查询记录到慢查询日志中。接下来,我们将基于慢查询日志中的一个例子进行分析。

假设有以下 SQL 查询是慢查询日志中的一条:

SELECT order_id, user_id, amount, order_date 
FROM orders 
WHERE user_id = 12345 AND order_date BETWEEN '2024-01-01' AND '2024-01-31'
ORDER BY order_date DESC;

二、使用 EXPLAIN 分析 SQL 查询

EXPLAIN 关键字可以帮助我们分析 SQL 执行计划,查看 MySQL 在执行查询时的处理方式,尤其是使用了哪些索引,如何扫描数据等。

1. 基本的 EXPLAIN 输出

通过 EXPLAIN 解析慢查询 SQL:

EXPLAIN SELECT order_id, user_id, amount, order_date 
FROM orders 
WHERE user_id = 12345 AND order_date BETWEEN '2024-01-01' AND '2024-01-31'
ORDER BY order_date DESC;

EXPLAIN 输出结果

idselect_typetabletypepossible_keyskeykey_lenrefrowsExtra
1SIMPLEordersALLidx_user_id, idx_order_dateNULLNULLNULL10000Using where; Using filesort
分析
  1. table:查询涉及的表是 orders
  2. type:扫描类型是 ALL,表示全表扫描。理想情况下,查询应该使用索引,而不是全表扫描。ALL 表示性能较差。
  3. possible_keys:显示 MySQL 可能使用的索引,包括 idx_user_ididx_order_date。这些索引覆盖了查询的字段。
  4. key:显示实际使用的索引,当前没有使用任何索引(NULL)。
  5. Extra:显示额外的信息,当前显示的是 Using where; Using filesort,表示使用了 WHERE 子句过滤数据,并且对查询结果进行了额外的文件排序。

EXPLAIN 的结果来看,MySQL 没有使用任何索引进行查询,而是进行全表扫描并在应用层进行排序。导致查询效率低下。


三、分析索引失效的原因

在上述查询中,尽管存在针对 user_idorder_date 的索引,MySQL 仍然没有使用这些索引。我们可以分析可能的原因:

1. WHERE 子句中有多个条件

WHERE 子句中有两个条件:user_id = 12345order_date BETWEEN '2024-01-01' AND '2024-01-31'。如果这两个条件中的字段没有组成联合索引,MySQL 可能选择不使用索引,导致全表扫描。

2. ORDER BY 子句影响

查询中包含 ORDER BY order_date DESC,这意味着 MySQL 在查询完成后需要对数据进行排序。如果没有合适的索引,MySQL 会使用 filesort,这也是 EXPLAIN 输出中的提示。

3. 索引选择性差

如果 user_idorder_date 字段的选择性较差(即某些值的重复程度较高),MySQL 可能认为索引的选择效率较低,反而选择进行全表扫描。


四、优化方案:

1. 创建联合索引

针对查询的 WHERE 子句和 ORDER BY 子句,我们可以创建一个联合索引,这样 MySQL 就可以同时使用索引来过滤数据和排序。

优化后的索引创建

CREATE INDEX idx_user_id_order_date ON orders(user_id, order_date);

这个索引可以同时支持 WHERE 条件中的 user_idorder_date 字段,并且能够优化 ORDER BY 操作。

2. 查看优化后的执行计划

创建索引后,我们可以再次使用 EXPLAIN 查看优化后的执行计划:

EXPLAIN SELECT order_id, user_id, amount, order_date 
FROM orders 
WHERE user_id = 12345 AND order_date BETWEEN '2024-01-01' AND '2024-01-31'
ORDER BY order_date DESC;

优化后的 EXPLAIN 输出

idselect_typetabletypepossible_keyskeykey_lenrefrowsExtra
1SIMPLEordersrefidx_user_id_order_dateidx_user_id_order_date5const100Using where; Using index
分析
  1. type:现在的类型是 ref,表示使用了索引。refALL 更高效,因为 MySQL 只扫描了相关的索引记录。
  2. key:使用了我们新创建的 idx_user_id_order_date 联合索引。
  3. Extra:显示了 Using where; Using index,意味着查询现在通过索引过滤数据,并且不需要额外的排序操作。

通过创建联合索引,查询性能得到了显著提升,MySQL 能够利用索引进行高效的数据过滤和排序。


五、进一步优化:使用 SHOW PROFILE 和 SHOW STATUS

除了 EXPLAIN,我们还可以使用 SHOW PROFILESHOW STATUS 进一步分析查询的性能瓶颈。

1. 使用 SHOW PROFILE

SHOW PROFILE 可以帮助我们查看 SQL 查询执行过程中的各个阶段,了解每个步骤的时间消耗。

SET PROFILING = 1;  -- 启用 Profiling
SELECT order_id, user_id, amount, order_date 
FROM orders 
WHERE user_id = 12345 AND order_date BETWEEN '2024-01-01' AND '2024-01-31'
ORDER BY order_date DESC;
SHOW PROFILE FOR QUERY 1;  -- 查看第一条查询的执行时间和资源消耗

这将显示每个阶段(如查询解析、排序、读取数据等)的时间消耗,帮助你发现瓶颈所在。

2. 使用 SHOW STATUS

SHOW STATUS 可以帮助我们了解当前数据库的状态,特别是在查询优化方面,我们可以查看相关的缓存命中率、索引使用情况等信息。

SHOW STATUS LIKE 'Handler_read_rnd_next';  -- 全表扫描次数
SHOW STATUS LIKE 'Handler_read_key';      -- 索引扫描次数

通过这些状态信息,我们可以监控查询优化的效果,确保索引被有效利用。


六、总结

  1. EXPLAIN 是优化查询的核心工具,它能帮助我们了解 MySQL 执行查询时的决策,分析索引是否有效、查询是否合理。
  2. 创建合适的索引:为查询条件和排序条件创建联合索引可以大幅提升查询性能,减少全表扫描。
  3. SHOW PROFILE 和 SHOW STATUS:这些工具提供了更详细的执行信息,可以帮助我们进一步识别和优化查询瓶颈。
  4. 索引优化是一项持续的工作,随着数据量的增加,可能需要不断调整索引和查询策略,以应对性能变化。

通过合理使用索引、优化 SQL 语句设计、分析执行计划,能够大幅提升 MySQL 查询性能,尤其是在处理复杂查询时,能有效减少性能瓶颈。

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

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

相关文章

【人工智能机器学习基础篇】——深入详解强化学习之常用算法Q-Learning与策略梯度,掌握智能体与环境的交互机制

深入详解强化学习之常用算法:Q-Learning与策略梯度 强化学习(Reinforcement Learning, RL)作为机器学习的一个重要分支,近年来在多个领域取得了显著成果。从棋类游戏的人机对战到自主驾驶汽车,强化学习技术展示了其强大…

计算机网络-L2TP VPN基础实验配置

一、概述 上次大概了解了L2TP的基本原理和使用场景,今天来模拟一个小实验,使用Ensp的网卡桥接到本地电脑试下L2TP拨号,今天主要使用标准的L2TP,其实在这个基础上可以加上IPSec进行加密,提高安全性。 拓扑说明&#xf…

Linux | 零基础Ubuntu解压RaR等压缩包文件

目录 介绍 案例分析 安装工具 解压实践 介绍 RAR是一种专利文件格式,用于数据压缩与归档打包,开发者为尤金罗谢尔(俄语:Евгений Лазаревич Рошал,拉丁转写:Yevgeny Lazarevich R…

Postman接口测试05|实战项目笔记

目录 一、项目接口概况 二、单接口测试-登录接口:POST 1、正例 2、反例 ①姓名未注册 ②密码错误 ③姓名为空 ④多参 ⑤少参 ⑥无参 三、批量运行测试用例 四、生成测试报告 1、Postman界面生成 2、Newman命令行生成 五、token鉴权(“…

PyTorch快速入门教程【小土堆】之完整模型训练套路

视频地址完整的模型训练套路(一)_哔哩哔哩_bilibili import torch import torchvision from model import * from torch import nn from torch.utils.data import DataLoader# 准备数据集 train_data torchvision.datasets.CIFAR10(root"CIFAR10&…

网络分析工具-tcpdump

文章目录 前言一、tcpdump基础官网链接命令选项详解常规过滤规则tcpdump输出 一、tcpdump实践HTTP协议ICMP状态抓包 前言 当遇到网络疑难问题的时候,抓包是最基本的技能,通过抓包才能看到网络底层的问题 一、tcpdump基础 tcpdump是一个常用的网络分析工…

可编辑31页PPT | 大数据湖仓一体解决方案

荐言分享:在当今数字化时代,大数据已成为企业决策和业务优化的关键驱动力。然而,传统的数据处理架构,如数据仓库和数据湖,各自存在局限性,难以满足企业对数据高效存储、灵活处理及实时分析的综合需求。因此…

Python中的sqlite3模块:SQLite数据库接口详解

Python中的sqlite3模块:SQLite数据库接口详解 主要功能sqlite3.connect(database)connection.cursor()cursor.execute(sql)connection.commit()cursor.fetchall()connection.close() 使用示例执行结果总结 在Python中,sqlite3模块提供了一个与SQLite数据…

easyui textbox使用placeholder无效

easyui textbox使用placeholder无效 在easyui 的textbox控件&#xff0c;请使用data-options 设定 示例 <input type text class easyui-textbox data-options "prompt:请输入您的邮箱"/>

[创业之路-232]:《华为闭环战略管理》-5-组织架构、业务架构、产品架构、技术架构、项目架构各自设计的原则是什么?

目录 一、组织架构设计原则 二、业务架构设计原则 三、产品架构设计原则 四、技术架构设计原则 五、项目架构设计原则 一、各自的组成元素 组织架构、业务架构、产品架构、技术架构、项目架构各自的组成元素具体如下&#xff1a; 组织架构 - 组织企业相似资源的方式&…

STM32中断详解

STM32中断详解 NVIC 中断系统中断向量表相关寄存器中断优先级中断配置 外部中断实验EXTI框图外部中断/事件线映射中断步骤初始化代码实现 定时器中断通用定时器相关功能标号1&#xff1a;时钟源标号 2&#xff1a;控制器标号 3&#xff1a;时基单元 代码实现 NVIC 中断系统 STM…

【LeetCode】200、岛屿数量

【LeetCode】200、岛屿数量 文章目录 一、并查集1.1 并查集1.2 多语言解法 二、洪水填充 DFS2.1 洪水填充 DFS 一、并查集 1.1 并查集 // go var sets int var father [90000]intfunc numIslands(grid [][]byte) int {n, m : len(grid), len(grid[0])build(grid, n, m)for i …

Github 2024-12-28 Rust开源项目日报 Top10

根据Github Trendings的统计,今日(2024-12-28统计)共有10个项目上榜。根据开发语言中项目的数量,汇总情况如下: 开发语言项目数量Rust项目10TypeScript项目1Python项目1egui: 一个简单、快速且高度可移植的 Rust GUI 库 创建周期:1903 天开发语言:Rust协议类型:Apache Li…

SOME/IP 协议详解——序列化

文章目录 0. 概述1.基本数据序列化2.字符串序列化2.1 字符串通用规则2.2 固定长度字符串规则2.3 动态长度字符串规则 3.结构体序列化4. 带有标识符和可选成员的结构化数据类型5. 数组5.1 固定长度数组5.2 动态长度数组5.3 Enumeration&#xff08;枚举&#xff09;5.4 Bitfield…

心力衰竭相关临床记录数据分析开发技术概述

心力衰竭相关临床记录数据分析开发技术概述 心力衰竭临床记录数据分析的开发涉及多种技术&#xff0c;包括数据采集、处理、建模和可视化等方面。以下是从技术角度对整个开发流程的概述&#xff1a; 数据采集技术 1.1 数据来源 公开数据集&#xff1a;如 UCI 数据存储库、Clin…

三分钟在你的react项目中引入tailwindcss

前言&#xff1a;在vite搭建的react项目中引入并使用tailwindcss 一、初始化react项目 1、创建项目 在文件夹下右键打开终端并输入命令使用vite创建项目 pnpm create vite react-tailwind选择reactjavascript&#xff0c;并输入命令安装依赖并启动 2、安装tailwind pnpm …

端到端性能体验稳定性优化常见方案

引言 作为一个服务端程序员&#xff0c;在一个领域深耕几年之后&#xff0c;会发现学习的新东西越来越少&#xff0c;这时候怎么办呢&#xff1f;就需要一边深耕&#xff0c;一边增加对相关横向团队的了解和交流&#xff0c;会发现原来端到端链路还有非常广阔的空间等着你去成长…

【AndroidAPP】权限被拒绝:[android.permission.READ_EXTERNAL_STORAGE],USB设备访问权限系统报错

一、问题原因 1.安卓安全性变更 Android 12 的安全性变更&#xff0c;Google 引入了更严格的 PendingIntent 安全管理&#xff0c;强制要求开发者明确指定 PendingIntent 的可变性&#xff08;Mutable&#xff09;或不可变性&#xff08;Immutable&#xff09;。 但是&#xf…

C之(14)gcov覆盖率

C之(14)gcov覆盖率 Author: Once Day Date: 2024年12月30日 一位热衷于Linux学习和开发的菜鸟&#xff0c;试图谱写一场冒险之旅&#xff0c;也许终点只是一场白日梦… 漫漫长路&#xff0c;有人对你微笑过嘛… 全系列文章可参考专栏: C语言_Once-Day的博客-CSDN博客 前些天…

简易屏幕共享工具-基于WebSocket

前面写了两个简单的屏幕共享工具&#xff0c;不过那只是为了验证通过截屏的方式是否可行&#xff0c;因为通常手动截屏的频率很低&#xff0c;而对于视频来说它的帧率要求就很高了&#xff0c;至少要一秒30帧率左右。所以&#xff0c;经过实际的截屏工具验证&#xff0c;我了解…