0

0

SQL 查询重写的系统方法

舞夢輝影

舞夢輝影

发布时间:2026-01-23 19:04:02

|

689人浏览过

|

来源于php中文网

原创

必须重写SQL而非加索引的情况包括:EXPLAIN显示全表扫描且rows值高,且WHERE含函数(如YEAR(created_at)=2023)、隐式类型转换(如user_id='123')或OR多非覆盖条件;此时应改函数为范围查询、避免隐式转换、拆分OR条件。

sql 查询重写的系统方法

什么时候必须重写 SQL 而不是加索引

索引解决不了所有性能问题。当 EXPLAIN 显示全表扫描 + 高 rows 值,且 WHERE 条件里有函数调用(如 WHERE YEAR(created_at) = 2023)、隐式类型转换(如 WHERE user_id = '123'user_idINT)、或 OR 连接多个非覆盖条件时,索引很可能失效——这时重写比调优更有效。

  • 函数包裹列 → 改成范围查询:WHERE created_at >= '2023-01-01' AND created_at
  • OR 多条件 → 拆成 UNION ALL(注意去重开销)或改用 IN(当语义等价时)
  • 隐式转换 → 统一数据类型:WHERE user_id = 123(而非字符串)

JOIN 顺序与驱动表选择的实际影响

MySQL 5.7+ 默认使用基于成本的优化器,但小表驱动大表仍是可靠原则。如果 EXPLAINtypeALLrows 极高,说明该表被当作驱动表,而它本身又没走索引——这是重写的明确信号。

  • 强制小表在前:把过滤后结果集最小的表放在 FROM 后第一个位置
  • 避免 LEFT JOIN 后再 WHERE 过滤右表字段(会退化为 INNER JOIN),应把条件移到 ON 子句
  • 多表 JOIN 时,用 STRAIGHT_JOIN(MySQL)临时固定顺序,验证效果后再决定是否保留

子查询转 JOIN 或窗口函数的取舍

相关子查询(correlated subquery)是性能黑洞,尤其在外部表数据量大时。但并非所有子查询都该转——要看数据库版本、数据分布和语义约束。

WaStar 网上花店系统
WaStar 网上花店系统

系统特点: 商品多级分类检索、搜索,支持同一商品多重分类,自由设置显示式样 自由设置会员类型,自由设置权限项目,自由分配每种会员类型和每个会员的权限 灵活的商品定价,最多12级价格自由分配给各种会员类型或会员,也可针对单会员单商品特殊定价 强大的会员管理、帐户管理、订单管理功能和一系列帐务查询统计功能 灵活的会员积分系统,自由设置每个积分事件的积分计算方法 灵活的网站内容发布、管理系统,每个栏目可

下载
  • 标量子查询(SELECT (SELECT name FROM users WHERE id = t.user_id))→ 优先转 JOIN,除非 users.id 不唯一需保语义
  • 存在性检查(WHERE EXISTS (SELECT 1 FROM logs l WHERE l.order_id = o.id))→ 可用 INNER JOIN 替代,但注意重复行;更稳的是保持 EXISTS 并确保 logs.order_id 有索引
  • MySQL 8.0+ / PostgreSQL 可用窗口函数替代部分聚合子查询,例如用 ROW_NUMBER() OVER (PARTITION BY category ORDER BY price DESC) 替代“每类最贵商品”类子查询

GROUP BY 和 HAVING 的隐藏开销点

GROUP BY 本身不慢,慢在分组键未索引、或 HAVING 中用了无法下推的表达式。如果 EXPLAIN 显示 Using temporary; Using filesort,且分组字段无索引,重写重点就在这里。

  • 确保 GROUP BY 字段有联合索引,顺序与分组顺序一致(如 GROUP BY dept, role → 索引应为 (dept, role)
  • HAVING COUNT(*) > 1 无法下推,但若能提前用 WHERE 过滤(如 WHERE status = 'active'),先缩小输入集再分组,效果远好于依赖 HAVING
  • 避免在 GROUP BY 中用函数或表达式(如 GROUP BY DATE(created_at)),改用生成列(MySQL 5.7+)或预计算字段

真正卡住的往往不是语法多复杂,而是没意识到某个 WHERE 条件本可以提前过滤掉 90% 的行——重写的第一步,永远是看懂执行计划里哪一行真正扫了最多数据。

相关专题

更多
数据分析工具有哪些
数据分析工具有哪些

数据分析工具有Excel、SQL、Python、R、Tableau、Power BI、SAS、SPSS和MATLAB等。详细介绍:1、Excel,具有强大的计算和数据处理功能;2、SQL,可以进行数据查询、过滤、排序、聚合等操作;3、Python,拥有丰富的数据分析库;4、R,拥有丰富的统计分析库和图形库;5、Tableau,提供了直观易用的用户界面等等。

685

2023.10.12

SQL中distinct的用法
SQL中distinct的用法

SQL中distinct的语法是“SELECT DISTINCT column1, column2,...,FROM table_name;”。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

323

2023.10.27

SQL中months_between使用方法
SQL中months_between使用方法

在SQL中,MONTHS_BETWEEN 是一个常见的函数,用于计算两个日期之间的月份差。想了解更多SQL的相关内容,可以阅读本专题下面的文章。

348

2024.02.23

SQL出现5120错误解决方法
SQL出现5120错误解决方法

SQL Server错误5120是由于没有足够的权限来访问或操作指定的数据库或文件引起的。想了解更多sql错误的相关内容,可以阅读本专题下面的文章。

1117

2024.03.06

sql procedure语法错误解决方法
sql procedure语法错误解决方法

sql procedure语法错误解决办法:1、仔细检查错误消息;2、检查语法规则;3、检查括号和引号;4、检查变量和参数;5、检查关键字和函数;6、逐步调试;7、参考文档和示例。想了解更多语法错误的相关内容,可以阅读本专题下面的文章。

359

2024.03.06

oracle数据库运行sql方法
oracle数据库运行sql方法

运行sql步骤包括:打开sql plus工具并连接到数据库。在提示符下输入sql语句。按enter键运行该语句。查看结果,错误消息或退出sql plus。想了解更多oracle数据库的相关内容,可以阅读本专题下面的文章。

717

2024.04.07

sql中where的含义
sql中where的含义

sql中where子句用于从表中过滤数据,它基于指定条件选择特定的行。想了解更多where的相关内容,可以阅读本专题下面的文章。

577

2024.04.29

sql中删除表的语句是什么
sql中删除表的语句是什么

sql中用于删除表的语句是drop table。语法为drop table table_name;该语句将永久删除指定表的表和数据。想了解更多sql的相关内容,可以阅读本专题下面的文章。

419

2024.04.29

C++ 高级模板编程与元编程
C++ 高级模板编程与元编程

本专题深入讲解 C++ 中的高级模板编程与元编程技术,涵盖模板特化、SFINAE、模板递归、类型萃取、编译时常量与计算、C++17 的折叠表达式与变长模板参数等。通过多个实际示例,帮助开发者掌握 如何利用 C++ 模板机制编写高效、可扩展的通用代码,并提升代码的灵活性与性能。

10

2026.01.23

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
MySQL 教程
MySQL 教程

共48课时 | 1.9万人学习

MySQL 初学入门(mosh老师)
MySQL 初学入门(mosh老师)

共3课时 | 0.3万人学习

简单聊聊mysql8与网络通信
简单聊聊mysql8与网络通信

共1课时 | 805人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号