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

news2025/1/5 7:13:28

在实际的开发过程中,随着数据量的不断增大,慢查询成为了不可忽视的性能瓶颈。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.coloradmin.cn/o/2269928.html

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

相关文章

TTL 传输中过期问题定位

问题: 工作环境中有一个acap的环境,ac的wan口ip是192.168.186.195/24,ac上lan上有vlan205,其ip子接口地址192.168.205.1/24,ac采用非nat模式,而是路由模式,在上级路由器上有192.168.205.0/24指向…

Cocos2dx Lua绑定生成中间文件时参数类型与源码类型不匹配

这两天维护的一个项目&#xff0c;使用arm64-v8a指令集编译时遇到了报错&#xff0c;提示类型不匹配&#xff0c;具体报错的代码【脚本根据C源文件生成的中间文件】如下&#xff1a; const google::protobuf::RepeatedField<unsigned long long>& ret cobj->equi…

连接Milvus

连接到Milvus 验证Milvus服务器正在侦听哪个本地端口。将容器名称替换为您自己的名称。 docker port milvus-standalone 19530/tcp docker port milvus-standalone 2379/tcp docker port milvus-standalone 192.168.1.242:9091/api/v1/health 使用浏览器访问连接地址htt…

走方格(蓝桥杯2020年试题H)

【问题描述】在平面上有一些二维点阵。这些点的编号就像二维数组的编号一样&#xff0c;从上到下依次为第1~n行&#xff0c;从左到右依次为第1~m列&#xff0c;每个点可以用行号和列号表示。 现在有个人站在第1行第1列&#xff0c;他要走到第n行第m列&#xff0c;只能向右或者向…

28. 二叉树遍历

题目描述 根据给定的二叉树结构描述字符串&#xff0c;输出该二叉树按照中序遍历结果字符串。中序遍历顺序为:左子树&#xff0c;根结点&#xff0c;右子树。 输入描述 由大小写字母、左右大括号、逗号组成的字符串: 1、字母代表一个节点值&#xff0c;左右括号内包含该节点的子…

Swift White Hawkstrider

Swift White Hawkstrider 迅捷白色陆行鸟 Swift White Hawkstrider - Item - 魔兽世界怀旧服TBC数据库_WOW2.43数据库_70级《燃烧的远征》数据库 Kaelthas Sunstrider (1) <Lord of the Blood Elves> 凯尔萨斯逐日者. 掉落 [80圣骑士][Alonsus-加丁][诺森德冒险补给品…

LeetCode算法题——有序数组的平方

题目描述 给你一个按非递减顺序排序的整数数组nums&#xff0c;返回每个数字的平方组成的新数组&#xff0c;要求也按非递减顺序排序。 题解 解法一&#xff1a;暴力解法 思路&#xff1a; 该题目可通过暴力解法解决&#xff0c;即利用for循环遍历数组&#xff0c;对数组每…

项目开发实践——基于SpringBoot+Vue3实现的在线考试系统(四)

文章目录 一、管理员角色功能实现1、添加教师功能实现1.1 页面设计1.2 前端功能实现1.3 后端功能实现1.4 效果展示2、教师管理功能实现2.1 页面设计2.2 前端功能实现2.3 后端功能实现2.3.1 后端查询接口实现2.3.2 后端编辑接口实现2.3.3 后端删除接口实现2.4 效果展示二、代码下…

DVWA靶场Brute Force (暴力破解) 漏洞low(低),medium(中等),high(高),impossible(不可能的)所有级别通关教程

目录 暴力破解low方法1方法2 mediumhighimpossible 暴力破解 暴力破解是一种尝试通过穷尽所有可能的选项来获取密码、密钥或其他安全凭证的攻击方法。它是一种简单但通常无效率的破解技术&#xff0c;适用于密码强度较弱的环境或当攻击者没有其他信息可供利用时。暴力破解的基…

基于feapder爬虫与flask前后端框架的天气数据可视化大屏

# 最近又到期末了&#xff0c;有需要的同学可以借鉴。 一、feapder爬虫 feapder是国产开发的新型爬虫框架&#xff0c;具有轻量且数据库操作方便、异常提醒等优秀特性。本次设计看来利用feapder进行爬虫操作&#xff0c;可以加快爬虫的速率&#xff0c;并且简化数据入库等操作…

抖音短视频矩阵系统源码开发技术解析

开发概览&#xff1a; 抖音短视频矩阵系统的构建基于一系列现代技术栈&#xff0c;主要包括VUE, Spring Boot和Django。本文档旨在为开发者提供关于短视频矩阵系统源代码的开发与部署指南。 技术框架分析&#xff1a; 前端技术选型&#xff1a; 对于前端界面的构建&#xf…

CentOS Stream 9 安装 JDK

安装前检查 java --version注&#xff1a;此时说明已安装过JDK&#xff0c;否则为未安装。如若已安装过JDK可以跳过安装步骤直接使用&#xff0c;或者先卸载已安装的JDK版本重新安装。 安装JDK 官网下载地址&#xff1a;https://www.oracle.com/java/technologies/downloads…

【git】git生成rsa公钥的方法

git生成rsa公钥的方法 一&#xff0c;简介二&#xff0c;操作方法三&#xff0c;总结 一&#xff0c;简介 在工作的过程中&#xff0c;经常需要生成rsa的密钥&#xff0c;然后提供给别人&#xff0c;然后别人给你开通代码下载权限。本文介绍如何在本地生成rsa的密钥供参考。 …

Solon 加入 GitCode:助力国产 Java 应用开发新飞跃

在当今数字化快速发展的时代&#xff0c;Java 应用开发框架不断演进&#xff0c;开发者们始终在寻找更快、更小、更简单的解决方案。近期&#xff0c;Solon 正式加入 GitCode&#xff0c;为广大 Java 开发者带来全新的开发体验&#xff0c;尤其是在国产应用开发进程中&#xff…

如何通过深度学习提升大分辨率图像预测准确率?

随着科技的不断进步&#xff0c;图像处理在各个领域的应用日益广泛&#xff0c;特别是在医疗影像、卫星遥感、自动驾驶、安防监控等领域中&#xff0c;大分辨率图像的使用已经成为了一项不可或缺的技术。然而&#xff0c;大分辨率图像带来了巨大的计算和存储压力&#xff0c;同…

硬件基础知识笔记(2)——二级管、三极管、MOS管

Part 2 二级管、三极管、MOS管 1、二级管1.1肖特基二极管和硅二极管选型比较1.2到底是什么决定了二极管的最高工作频率&#xff1f;1.3二极管结电容和反向恢复时间都是怎么来的1.4肖特基二极管的工作原理1.5为什么要用肖特基二极管续流&#xff1f; 2、三极管2.1三极管工作原理…

操作系统论文导读(八):Schedulability analysis of sporadic tasks with multiple criticality specifications——具有多个

Schedulability analysis of sporadic tasks with multiple criticality specifications——具有多个关键性规范的零星任务的可调度性分析 目录 一、论文核心思想 二、基本定义 2.1 关键性指标 2.2 任务及相关参数定义 2.3 几个基础定义 三、可调度性分析 3.1 调度算法分…

word中文献引用[]符号的上下标格式修改

word中文献引用[]符号的上下标格式修改 百度网址 1、查找打开使用通配符&#xff0c;输入[[][0-9]{1,2}[]]&#xff0c;即可匹配所有的字[1],[12]这些字符&#xff0c;然后鼠标点击替换为的空白处&#xff0c;再点击特殊格式–>“字体”&#xff0c;选中上标&#xff0c;最…

在 ArcGIS Pro/GeoScene Pro 中设计专题地图的符号系统

原始 按颜色对面进行符号化 打开符号系统 选择主符号系统 选择字段及其计算方式 更改临界值</

_使用CLion的Vcpkg安装SDL2,添加至CMakelists时报错,编译报错

语言&#xff1a;C20 编译器&#xff1a;gcc 14.2 摘要&#xff1a;初次使用Vcpkg添加SDL2&#xff0c;出现CMakelists找不到错误、编译缺失main错误、运行失败错误。 CMakelists缺失错误&#xff1a; 使用CLion的Vcpkg安装SDL2时&#xff0c;按照指示把对应代码添加至CMakel…