MySQL中如何进行SQL调优举例详解

 更新时间:2025年01月29日 08:54:10   作者:web2u  
这篇文章主要介绍了SQL调优的几种方法,包括合理设计索引,避免SELECT*,避免在SQL中进行函数计算等操作,避免使用%LIKE,注意联合索引需满足最左匹配原则,不要对无索引字段进行排序操作,文中通过代码介绍的非常详细,需要的朋友可以参考下

重点

平时进行 SQL 调优,主要是通过观察慢 SQL,然后利用 explain 分析查询语句的执行计划,识别性能瓶颈,优化查询语句。

1) 合理设计索引,利用联合索引进行覆盖索引的优化,避免回表的发生,减少一次查询和随机 I/O

  • 回表:索引无法满足查询所需的所有列数据,需要回到主表获取额外的数据。
  • 避免回表:创建覆盖索引(索引包含了查询所需的所有列),让查询可以直接从索引中获取所有数据,无需访问主表。

例子:

建表和建立索引:

CREATE TABLE user (
    id INT PRIMARY KEY,
    name VARCHAR(50),
    age INT,
    gender CHAR(1),
    city VARCHAR(50)
);
CREATE INDEX idx_name_age_gender ON user(name, age, gender);  
  • 建立了联合索引:nameagegender

若执行SELECT city FROM user WHERE name = 'John' AND age = 25; 因为 select 需要 返回city。 索引中没有city列的数据,还需要根据索引条目中包含的主键信息(虽然例子中没有显式指定,但通常索引会包含指向主键的指针)回到 user 表的主键索引中,去查找完整的行数据,这个“回到主表查找 city 列”的过程就是回表 。

2) 避免 SELECT *,只查询必要的字段

3) 避免在 SQL 中进行函数计算等操作,使得无法命中索引

4) 避免使用 %LIKE,导致全表扫描

5) 注意联合索引需满足最左匹配原则

解释最左匹配原则:最左匹配原则是指在使用联合索引时,必须按照索引的顺序从左到右使用,不能跳过索引中的列。
1. SQL 实战理解 最左匹配原则建表语句:假设我们有一个用户订单表,包含用户ID、订单日期和订单金额三个字段,我们对这三个字段创建一个联合索引。

CREATE TABLE user_orders (
    id INT AUTO_INCREMENT PRIMARY KEY,
    user_id INT,
    order_date DATE,
    order_amount DECIMAL(10,2),
    INDEX idx_user_date_amount (user_id, order_date, order_amount)
);

Python脚本生成测试数据:

from datetime import datetime, timedelta
import random

# 生成INSERT语句
def generate_insert_statements():
    start_date = datetime(2023, 1, 1)
    statements = []
    
    for _ in range(4200):
        user_id = random.randint(1, 1000)
        days = random.randint(0, 365)
        order_date = (start_date + timedelta(days=days)).strftime('%Y-%m-%d')
        order_amount = round(random.uniform(10.0, 1000.0), 2)
        
        insert_sql = f"INSERT INTO user_orders (user_id, order_date, order_amount) VALUES ({user_id}, '{order_date}', {order_amount});"
        statements.append(insert_sql)
    
    # 将所有INSERT语句写入文件
    with open('insert_data.sql', 'w') as f:
        f.write('\n'.join(statements))
        
    print("INSERT语句已生成到 insert_data.sql 文件中")

if __name__ == "__main__":
    generate_insert_statements()

测试不同查询场景:

-- 完全满足最左匹配原则(使用全部索引列)
EXPLAIN SELECT * FROM user_orders 
WHERE user_id = 100 AND order_date = '2023-05-01' AND order_amount = 500;

-- 满足最左匹配原则(使用索引的前两列)
EXPLAIN SELECT * FROM user_orders 
WHERE user_id = 100 AND order_date = '2023-05-01';

explain 结果:

-- 满足最左匹配原则(只使用第一列)
EXPLAIN SELECT * FROM user_orders 
WHERE user_id = 100;

explain 结果:

-- 不满足最左匹配原则(跳过`user_id`)
EXPLAIN SELECT * FROM user_orders 
WHERE order_date = '2023-05-01' AND order_amount = 500;

explain 结果:

-- 不满足最左匹配原则(只使用order_date)
EXPLAIN SELECT * FROM user_orders 
WHERE order_date = '2023-05-01';

explain 结果:

-- 不满足最左匹配原则(只使用order_amount)
EXPLAIN SELECT * FROM user_orders 
WHERE order_amount = 500;

explain 结果:

  • 从上述explain 的结果看出,不满足最左匹配原则,filitered 都很低。

6) 不要对无索引字段进行排序操作

  • 强制使用文件排序(filesort):
    当对无索引字段排序时,MySQL无法利用索引的有序性,必须将数据加载到内存中进行排序,这就是filesort,filesort是一个非常耗费资源的操作。

  • 内存开销大
    如果排序数据量小,MySQL会在内存中完成排序,如果数据量超过sort_buffer_size,会发生磁盘文件排序,磁盘排序涉及临时文件的创建和多次IO,性能更差!

SQL实战演示

-- 创建测试表
CREATE TABLE worker(
    id INT AUTO_INCREMENT PRIMARY KEY,
    name VARCHAR(50),
    salary DECIMAL(10,2),
    department VARCHAR(50),
    INDEX idx_salary (salary)  -- 只对salary创建索引
);

-- 插入测试数据
INSERT INTO worker(name, salary, department) VALUES
('张三', 5000, '技术部'),
('李四', 6000, '市场部'),
('王五', 4500, '技术部'),
('赵六', 7000, '销售部');
-- 会使用索引排序的情况:
-- 只查询索引列
SELECT salary FROM employees ORDER BY salary;
-- 或者
SELECT id, salary FROM employees ORDER BY salary;
-- 结果显示: Using index for order by

会导致filesort的情况:
-- 特例:查询所有列(SELECT *)
SELECT * FROM employees ORDER BY salary;

  • 当使用SELECT *时,需要回表获取所有列的数据,这种情况下,MySQL认为使用索引排序的成本比filesort更高。
-- 对无索引的department字段排序
EXPLAIN SELECT * FROM employees ORDER BY department;
-- 结果显示: Using filesort

7) 连表查询需要注意不同字段的字符集是否一致,否则也会导致全表扫描

除此之外,还可以利用缓存来优化,一些变化少或者访问频繁的数据设置到缓存中,减轻数据库的压力,提升查询的效率。

还可以通过业务来优化,例如少展示一些不必要的字段,减少多表查询的情况,将列表查询替换成分页分批查询等等。

总结

到此这篇关于MySQL中如何进行SQL调优的文章就介绍到这了,更多相关MySQL进行SQL调优内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 一文详解MySQL单表存多大的数据量比较合适

    一文详解MySQL单表存多大的数据量比较合适

    MySQL单表的数据量上限主要受存储引擎和文件系统限制,理论值可达数千万至数十亿行,但实际建议控制在千万级以内以保证性能,这篇文章主要介绍了MySQL单表存多大的数据量比较合适的相关资料,需要的朋友可以参考下
    2026-02-02
  • Linux下安装MySQL5.7.19问题小结

    Linux下安装MySQL5.7.19问题小结

    第一次在自己虚机上安装mysql 中间碰到很多问题 在这里记下来,特此分享到脚本之家平台供大家参考
    2017-08-08
  • mysql数据库的全量与增量的备份以及恢复方式

    mysql数据库的全量与增量的备份以及恢复方式

    在数据库管理中,全量备份与恢复是将整个数据库的数据导出并在需要时完整地恢复,这通常使用mysqldump工具完成,增量备份则是在全量备份的基础上,只备份那些自上次全量备份后发生变化的数据,这需要数据库的二进制日志(binlog)开启
    2024-09-09
  • MySQL数据库索引和事务图文详解

    MySQL数据库索引和事务图文详解

    在当今数据驱动的时代,数据库的高效与可靠性是业务系统的核心支柱,而索引和事务作为数据库的两大基石,直接影响着数据查询性能与操作安全性,这篇文章主要介绍了MySQL数据库索引和事务的相关资料,需要的朋友可以参考下
    2026-01-01
  • 选择MySQL数据库进行连接的简单示例

    选择MySQL数据库进行连接的简单示例

    这篇文章主要介绍了选择MySQL数据库进行连接的简单示例,是MySQL入门学习中的基础知识,需要的朋友可以参考下
    2015-05-05
  • MySQL进阶查询、聚合查询和联合查询

    MySQL进阶查询、聚合查询和联合查询

    这篇文章主要介绍了MySQL数据库的进阶查询,聚合查询及联合查询,文中有详细的代码示例,需要的朋友可以参考阅读
    2023-04-04
  • Python MySQL进行数据库表变更和查询

    Python MySQL进行数据库表变更和查询

    这篇文章主要介绍了Python MySQL进行数据库表变更和查询的相关资料,需要的朋友可以参考下
    2017-05-05
  • MySQL通过show status查看、explain分析优化数据库性能

    MySQL通过show status查看、explain分析优化数据库性能

    这篇文章介绍了MySQL通过show status查看、explain分析优化数据库性能的方法,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-04-04
  • MySQL查询本周、上周、本月、上个月份数据的sql代码

    MySQL查询本周、上周、本月、上个月份数据的sql代码

    MySQL查询的方式很多,下面为您介绍的MySQL查询实现的是查询本周、上周、本月、上个月份的数据,如果您对MySQL查询方面感兴趣的话,不妨一看
    2012-11-11
  • MySQL中Stmt 预处理提高效率问题的小研究

    MySQL中Stmt 预处理提高效率问题的小研究

    在oracle数据库中,有一个变量绑定的用法,很多人都比较熟悉,可以调高数据库效率,应对高并发等,好吧,这其中并不包括我,当同事问我MySQL中有没有类似的写法时,我是很茫然的,于是就上网查,找到了如下一种写法
    2011-08-08

最新评论