MySQL 8.4 分区表实战:为什么分了区还是全分区扫描
来源:17golang MySQL频道原创
时间:2026-06-04 13:50:37 133浏览 收藏
先说结论:分区表快不快,先看有没有发生分区裁剪
我见过不少项目把大表一分区,就以为慢查询自然会消失。结果上线后慢日志还是刷,甚至比单表更难看。原因很简单:MySQL 分区表真正能帮你的前提,是优化器能通过查询条件排除无关分区,也就是 partition pruning。裁剪失败时,分区表可能变成“多个小表一起扫”。
本文用订单历史表举例,重点讲 MySQL 8.x 里分区裁剪如何验证、哪些 SQL 写法会失效、分区内索引怎么配合,以及上线前我会怎么检查。

业务场景:订单表按月分区后仍然慢
假设订单表按支付时间做月度 RANGE COLUMNS 分区:
CREATE TABLE orders (
id BIGINT NOT NULL,
user_id BIGINT NOT NULL,
paid_at DATETIME NOT NULL,
status TINYINT NOT NULL,
amount DECIMAL(12,2) NOT NULL,
PRIMARY KEY (id, paid_at),
KEY idx_user_paid (user_id, paid_at)
) ENGINE=InnoDB
PARTITION BY RANGE COLUMNS(paid_at) (
PARTITION p202605 VALUES LESS THAN ('2026-06-01'),
PARTITION p202606 VALUES LESS THAN ('2026-07-01'),
PARTITION p202607 VALUES LESS THAN ('2026-08-01'),
PARTITION pmax VALUES LESS THAN (MAXVALUE)
);
按理说,查 6 月订单应该只访问 p202606。但线上慢 SQL 常常写成这样:
SELECT id, amount FROM orders WHERE DATE(paid_at) = '2026-06-01' AND status = 1 ORDER BY id DESC LIMIT 100;
问题在 DATE(paid_at)。分区键被函数包裹后,优化器很难在执行前判断到底哪些分区一定不需要访问。你以为按天查,实际可能多个分区都被扫一遍。
诊断第一步:EXPLAIN PARTITIONS
排分区表慢查询时,我第一眼看 EXPLAIN PARTITIONS,不是先看普通索引。
EXPLAIN PARTITIONS SELECT id, amount FROM orders WHERE DATE(paid_at) = '2026-06-01' AND status = 1 ORDER BY id DESC LIMIT 100;
如果 partitions 里出现了很多个月份,说明分区裁剪失败。此时就算 key 看起来用了某个索引,也只是每个被访问分区内各扫一遍,整体成本仍然可能很高。

正确写法:把条件写成分区键可推导的范围
把函数条件改成半开区间,通常更利于裁剪:
SELECT id, amount FROM orders WHERE paid_at >= '2026-06-01' AND paid_at
再次执行:
EXPLAIN PARTITIONS SELECT id, amount FROM orders WHERE paid_at >= '2026-06-01' AND paid_at
这时候理想结果是 partitions 只包含 p202606。如果仍然访问多个分区,就继续查类型转换、时区边界、OR 条件和分区定义是否匹配。

踩坑原因:这些写法很容易让裁剪失效
- 函数包裹分区键。 比如
DATE(paid_at)、YEAR(paid_at)、TO_DAYS(paid_at)包在查询侧,容易让裁剪变差。 - 隐式类型转换。 分区键是日期或整数,条件却传入格式不稳定的字符串,优化器判断会变复杂。
- 跨分区键的 OR。
paid_at条件和其他字段用 OR 拼在一起,常常会扩大访问分区。 - 只建分区不建索引。 裁剪只负责少访问分区,进入目标分区以后仍然要靠普通索引完成过滤、排序和回表控制。
分区内索引也要重新设计
分区不是索引替代品。比如常见查询是按用户查近 30 天订单:
SELECT id, paid_at, amount FROM orders WHERE user_id = 88001 AND paid_at >= '2026-06-01' AND paid_at
这类 SQL 既需要分区裁剪,也需要分区内的 (user_id, paid_at) 索引。没有后者,优化器即使只进一个月分区,也可能在目标分区里扫大量用户数据。
上线检查:分区表最怕没人管未来分区
我上线分区表一定会检查三件运维事。第一,未来分区有没有提前创建,避免数据落入 pmax 后难清理。第二,归档删除是不是用 DROP PARTITION 或 TRUNCATE PARTITION 这类明确动作,而不是大事务 delete。第三,慢 SQL 监控里要带上访问分区数,至少定期抽样 EXPLAIN PARTITIONS。
ALTER TABLE orders
ADD PARTITION (
PARTITION p202608 VALUES LESS THAN ('2026-09-01')
);
如果业务已经大量写入 pmax,不要直接在线上硬拆,先评估数据量、复制延迟、DDL 影响和回滚方案。
个人经验:分区解决的是管理和裁剪,不是所有慢查询
分区表最适合两类收益:按时间快速清理历史数据,以及让明确时间范围查询少访问分区。它不适合被当成“万能大表优化按钮”。查询条件不带分区键、排序无法利用索引、热点集中在当前分区,这些问题不会因为分区自动消失。
我的做法是:先拿线上 TOP SQL 反推分区键;确认 80% 以上核心查询都能带上分区范围;再设计分区内索引和运维脚本。只为了“表太大看着不舒服”而分区,后面往往会付出更高维护成本。
总结
MySQL 8.x 分区表要想真正变快,必须同时满足三点:查询条件能触发分区裁剪;目标分区内有合适索引;分区生命周期有人维护。排查时先看 EXPLAIN PARTITIONS 的 partitions 字段,再看 key 和 rows。如果分区裁剪都没发生,别急着怪索引,先把 SQL 写法和分区键条件改正确。
-
数据库 · MySQL | 2天前 | 执行计划 · MySQL教程 · 慢查询治理 · 索引优化 · 数据库实战 · mysql 执行计划 慢查询 索引优化 MySQL 8 EXPLAIN ANALYZE389 收藏
-
数据库 · MySQL | 2天前 | InnoDB · MySQL教程 · 数据库实战 · 死锁排查 · 锁等待 · mysql innodb 死锁 事务 锁等待 MySQL 8 data_locks105 收藏
-
数据库 · MySQL | 1天前 | MySQL教程 · 数据库实战 · 在线DDL · ALTER TABLE · 元数据锁 · mysql innodb MySQL 8 在线 DDL ALTER TABLE MDL 元数据锁 INSTANT323 收藏
-
数据库 · MySQL | 1天前 | 性能优化 · InnoDB · 生产实践 · MySQL教程 · 数据库运维 · mysql redo log innodb 性能优化 innodb_redo_log_capacity382 收藏
-
388 收藏
-
数据库 · MySQL | 6分钟前 | binlog · 故障恢复 · 备份恢复 · MySQL教程 · DBA实战 · mysql DBA binlog 备份恢复 mysqlbinlog MySQL 8.4 PITR432 收藏
-
数据库 · MySQL | 27分钟前 | 字符集 · 故障排查 · MySQL教程 · 索引优化 · 排序规则 · mysql 排序规则 索引优化 utf8mb4 collation MySQL 8.4294 收藏
-
数据库 · MySQL | 37分钟前 | binlog · 主从复制 · 故障排查 · MySQL教程 · DBA实战 · mysql DBA binlog 主从复制 MySQL 8.4 复制延迟 relay log119 收藏
-
数据库 · MySQL | 1小时前 | 高并发 · 故障排查 · MySQL教程 · 事务隔离 · InnoDB锁 · mysql innodb 高并发 锁等待 MySQL 8.4 NOWAIT SKIP LOCKED439 收藏
-
数据库 · MySQL | 4小时前 | MySQL教程 · 慢查询治理 · 索引优化 · JSON查询 · InnoDB实战 · mysql JSON 慢查询 索引优化 MySQL 8.4 多值索引291 收藏
-
数据库 · MySQL | 23小时前 | InnoDB · 故障排查 · 生产实践 · MySQL教程 · 事务隔离 · mysql innodb Purge Lag History List 长事务 Undo326 收藏
-
数据库 · MySQL | 1天前 | 性能优化 · 执行计划 · 生产实践 · MySQL教程 · 索引优化 · mysql explain 索引优化 Index Condition Pushdown ICP179 收藏
-
189 收藏
-
数据库 · MySQL | 1天前 | 性能优化 · 执行计划 · 生产实践 · MySQL教程 · 数据库运维 · mysql 直方图 EXPLAIN ANALYZE Histogram 优化器统计信息419 收藏
-
388 收藏
-
数据库 · MySQL | 1天前 | 性能优化 · InnoDB · 生产实践 · MySQL教程 · 数据库运维 · mysql redo log innodb 性能优化 innodb_redo_log_capacity382 收藏
-
数据库 · MySQL | 1天前 | MySQL教程 · 数据库实战 · 在线DDL · ALTER TABLE · 元数据锁 · mysql innodb MySQL 8 在线 DDL ALTER TABLE MDL 元数据锁 INSTANT323 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习