MySQL:SQL 慢查询优化的技术指南

1、简述

在 Java 后端开发中,数据库是系统性能瓶颈的高发地带,而 慢 SQL 查询 往往是系统响应迟缓的“罪魁祸首”。本文将全面梳理慢 SQL 的优化思路,并结合 Java 示例进行实战演练。

在这里插入图片描述


2、慢查询的常见表现

慢查询通常表现为:

  • 接口响应时间缓慢
  • 数据库 CPU 占用高
  • 表锁、死锁频繁
  • Java 应用线程池阻塞严重

慢 SQL 的主要成因

成因类型说明
未使用索引全表扫描,查询耗时
使用了低效的函数或表达式LIKE '%xx%', DATE()
多表关联不当join 条件缺失或不走索引
过多返回字段只用到了部分字段却 SELECT *
where 条件不精准无法过滤大量无关数据
数据库设计不合理字段冗余、缺乏范式、字段类型错误等

3、慢查询优化的通用思路

✅ 加索引(重点)

  • WHEREJOINORDER BYGROUP BY 中涉及的字段加索引
  • 避免使用函数包裹字段,如 LEFT(name, 3),会导致无法使用索引

✅ 使用 EXPLAIN 分析执行计划

EXPLAIN SELECT * FROM orders WHERE user_id = 100;

关注字段:

字段说明
type连接类型(越接近 const 越好)
rows扫描行数(越小越好)
key使用的索引名称
Extra是否使用临时表、排序等

✅ 分页优化

避免深度分页:

-- 慢查询(跳过大量行)
SELECT * FROM orders LIMIT 1000000, 20;-- 推荐(使用上次主键记录)
SELECT * FROM orders WHERE id > 1000000 LIMIT 20;

✅ 拆表分区

  • 垂直拆分:将大表按字段拆分为多个表
  • 水平分表:按业务字段分库分表(如 user_id 分表)
  • 分区表:MySQL 原生支持(适合历史归档数据)

✅ 减少嵌套子查询

使用 JOIN 或临时表替代子查询,更高效。

✅ SQL 只查需要的字段

-- 慎用
SELECT * FROM user;-- 推荐
SELECT id, name, email FROM user;

4、慢 SQL 实践排查与优化

✅ 示例:慢查询前后对比

原始 SQL(慢)
SELECT * FROM orders WHERE DATE(create_time) = '2024-01-01';

📉 问题:

  • 使用了 DATE() 函数,索引失效
  • 全表扫描,耗时严重
优化 SQL(快)
SELECT * FROM orders 
WHERE create_time >= '2024-01-01 00:00:00' AND create_time < '2024-01-02 00:00:00';

📈 优点:

  • 范围查询走索引
  • 支持时间范围过滤

✅ Java 中日志配置监控慢 SQL

# application.yml 示例(Spring Boot)
logging:level:com.zaxxer.hikari.HikariConfig: DEBUGcom.zaxxer.hikari: TRACE
spring:datasource:url: jdbc:mysql://localhost:3306/demousername: rootpassword: roothikari:maximum-pool-size: 10connection-timeout: 3000

使用工具(如 p6spy)打印 SQL 及耗时,或开启 MySQL 慢查询日志:

-- MySQL 开启慢查询日志
SET GLOBAL slow_query_log = 'ON';
SET GLOBAL long_query_time = 1; -- 超过 1 秒记录

✅ SQL 优化 checklist

  • 是否使用了合适的索引
  • 是否避免了函数、表达式阻碍索引
  • 是否使用了 EXPLAIN 检查执行计划
  • 是否合理分页、避免深度翻页
  • 是否控制了查询字段数量
  • 是否考虑拆分大表或分区表
  • 是否避免了嵌套子查询

5、SQL 优化实战样例

场景 1:模糊查询优化

-- 慢:前置通配符无法使用索引
SELECT * FROM user WHERE name LIKE '%abc%';-- 优化:使用全文索引或右模糊匹配
SELECT * FROM user WHERE name LIKE 'abc%';

场景 2:避免函数阻碍索引

-- 慢
SELECT * FROM orders WHERE YEAR(create_time) = 2024;-- 快
SELECT * FROM orders 
WHERE create_time >= '2024-01-01' AND create_time < '2025-01-01';

场景 3:多字段组合索引使用顺序

-- 有联合索引 (user_id, status)-- 推荐:user_id 和 status 都参与
SELECT * FROM orders WHERE user_id = 100 AND status = 'PAID';-- 不推荐:只用 status,索引无法生效
SELECT * FROM orders WHERE status = 'PAID';

6、结语

慢查询是系统性能优化的重要战场。对于 Java 开发者而言,理解 SQL 执行机制和优化原则,比“用缓存”更根本、更有效

日常开发中,应做到:

  • 编写 SQL 前先考虑是否能走索引
  • 查询慢时第一时间用 EXPLAIN 排查
  • 数据库设计时就考虑查询结构

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

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

相关文章

leetcode543-二叉树的直径

leetcode 543 思路 路径长度计算&#xff1a;任意两个节点之间的路径长度&#xff0c;等于它们的最低公共祖先到它们各自的深度之和递归遍历&#xff1a;通过后序遍历&#xff08;左右根&#xff09;计算每个节点的左右子树深度&#xff0c;并更新全局最大直径深度与直径的关…

详解main的参数并实现读取文件

在 C 语言中&#xff0c;main函数的参数argc和argv用于接收命令行传入的参数 main 函数的两个参数 int main(int argc, char* argv[]) 假设顾客通过手机 APP 点餐&#xff0c;订单信息会被传递给餐厅的处理系统&#xff08;也就是你的程序&#xff09;。 订单信息结构 argc…

c++IO类

概述 c不直接处理输入输出&#xff0c;而是通过定义在标准类库中的类来处理IO。这些类支持从设备读取数据&#xff0c;向设备写入数据的IO操作&#xff0c;设备可以是文件、控制台窗口等。还可以从内存IO。 IO类 iostream: istream&#xff0c;wistreamostream&#xff0c;wo…

springboot的后端处理HTML的页面请求

下面是一个完整的 Spring Boot 后端示例&#xff0c;用于接收 <form> 提交的文件上传请求&#xff08;/article/uploadLifeImage 接口&#xff09;&#xff0c;并将上传的文件保存到本地目录。 ✅ 一、项目结构 upload-demo/ ├── src/ │ └── main/ │ ├…

深入探究 Go 语言中使用 SQLite 数据库

引言 在软件开发中&#xff0c;数据库是管理和存储数据的关键组件。SQLite 作为一款轻量级的嵌入式数据库&#xff0c;因其零配置、高性能和易于集成等特性&#xff0c;成为众多小型项目和嵌入式系统的理想选择。而 Go 语言以其高效、简洁的特点&#xff0c;为操作 SQLite 数据…

Portable Computer Power Adapter

Portable Computer Power Adapter 笔记本电源适配器&#xff0c;将220伏特的交流电转化直流电 现在的适配器真的体积之大&#xff0c;让我无法理解&#xff0c;本来便携计算机为了方便减少体积重量&#xff0c;现在都倒反天罡了。让我无法理解设计师是怎么干出来的。这玩意有2…

Uniapp 网络请求封装专题

目录 一、前言 二、uniapp官方文档 三、举例演示 3.1 使用说明 3.2 Content-Type 3.2.1 ​​基本概念 ​​3.2.2 核心作用 3.2.3 常见 Content-Type 类型及使用场景 1&#xff09;文本类 a&#xff09;text/plain​​​​ b&#xff09;text/html​​ 2&#xf…

2025年渗透测试面试题总结-2025年HW(护网面试) 07(题目+回答)

安全领域各种资源&#xff0c;学习文档&#xff0c;以及工具分享、前沿信息分享、POC、EXP分享。不定期分享各种好玩的项目及好用的工具&#xff0c;欢迎关注。 目录 2025年HW(护网面试) 07 一、OWASP Top 10 2023核心漏洞 二、XSS窃取Cookie全流程 三、渗透测试五阶段模型…

Seata分布式事务解决框架

Seata&#xff08;Simple Extensible Autonomous Transaction Architecture&#xff09;是一个开源的分布式事务解决方案&#xff0c;旨在帮助开发者更容易地在微服务架构中解决分布式事务问题。 你可以把它理解为一个工具箱&#xff0c;专门用来处理微服务之间操作的一致性。…

旧物回收小程序开发:开启绿色生活新方式

在环保理念日益深入人心的今天&#xff0c;每一件旧物都承载着资源再生的无限可能。我们精心打造的旧物回收小程序&#xff0c;宛如一把神奇的钥匙&#xff0c;为你开启绿色生活新方式&#xff01; 想象一下&#xff0c;家中堆积如山的旧衣物、闲置的电子产品、废弃的书籍杂志…

STM32 串口通信②:蓝牙模块HC-05控制单片机

一 前言 上一篇我们已经成功实现单片机和电脑的连接&#xff0c;接下来&#xff0c;我们学习一个有趣的板块&#xff0c;HC-05蓝牙模块&#xff0c;这个蓝牙模块&#xff0c;我们就要建立手机和单片机的通讯啦&#xff0c;还是比较有趣的一个过程&#xff0c;大家可以跟着多操作…

【Verilog】Verilator的TestBench该用C++还是SystemC

Verilator的Testbench&#xff08;测试平台&#xff09;主要使用 C 或 SystemC 来编写。这是由Verilator的工作原理决定的&#xff1a;它将你的Verilog/SystemVerilog设计转换成一个C类&#xff0c;因此你需要一个C环境来实例化和驱动这个类。 下面详细说明这两种方式以及如何…

71-Oracle Undo与Flashback管理(Guarantee设置)深度解析

小伙伴们有没有在操作undo表空间、Flashback管理&#xff0c;时间保留设置&#xff0c;总有些配置需要提前预置好&#xff0c;否则闪回查询和表的时候出现报错。 需短期恢复&#xff08;秒级~小时&#xff09;直接上UNDO_RETENTION同时启用RETENTION GUARANTEE&#xff0c;Und…

单片机——浮点数转换4位数码管显示

浮点数转换4位数码管显示 static char buffer[5]; int DecimalPlace 0; #define HideChar h void DisplayFloatOn4LED(float value) {long integer roundf(value );if (integer > 9999) //4位{integer 9999;snprintf(buffer, sizeof(buffer), "%4ld", integer…

金融行业B端系统布局实战:风险管控与数据可视化的定制方案

摘要 在金融行业&#xff0c;一次小小的数据泄露可能引发千万级资金损失&#xff0c;一次错误的风险评估或许让企业陷入危机。传统 B 端系统布局在应对复杂多变的金融业务时&#xff0c;常出现风险预警滞后、数据杂乱无章的情况&#xff0c;让从业者如履薄冰。如何才能在瞬息万…

融合LSTM与自注意力机制的多步光伏功率预测新模型解析

这篇论文《Improved multistep ahead photovoltaic power prediction model based on LSTM and self-attention with weather forecast data》&#xff08;2024, Applied Energy&#xff09;聚焦在 多步光伏功率预测 中&#xff0c;如何结合 LSTM 与自注意力机制&#xff08;se…

Blazor-内置输入组件

封装的输入组件 InputCheckbox&#xff1a;表示复选框。InputDate&#xff1a;表示类型为 date 的日期选择框。InputFile&#xff1a;表示文件上传。InputNumber&#xff1a;表示数字框。InputRadio&#xff1a;表示单选按钮。InputRadioGroup&#xff1a;表示单选按钮组。Inpu…

Qt源码分析: QChildEvent

本文记录QChildEvent事件相关代码分析。 注1&#xff1a;限于笔者研究水平&#xff0c;难免有表述不当&#xff0c;欢迎批评指正。 注2&#xff1a;博文会不定期更新&#xff0c;敬请关注。 一、QChildEvent的发送 分析QObject::setParent代码&#xff0c;当修改父对象时&…

《Whisper :说明书 》

[论文] [模型卡] [Colab 示例] Whisper 是一种通用的语音识别模型。它基于各种音频的大型数据集进行训练&#xff0c;也是一种多任务模型&#xff0c;可以执行多语言语音识别、语音翻译和语言识别。 方法 Transformer 序列到序列模型针对各种语音处理任务进行训练&#xff0c;…

回溯----8.N皇后

题目链接 /** 将n个棋子放在n*n的棋盘上,不同列,不同行,不同斜线 大致执行流程: 首先选取第一行第一格放置第一个棋子,再从第二行第一个位置开始选取合法的位置(不同行不同列不同斜线)放置棋子,重复上述流程迭代行数, 直到放置n个棋子。 若放置途中出现无合法位置的情况,回溯将…