MySQL以16KB页为单位从磁盘加载数据到InnoDB缓冲池,未命中则同步读整页;WHERE条件优先在存储引擎层用索引过滤,无法下推的表达式由Server层二次过滤;SELECT列指定可避免读溢出页,LIMIT仅在能索引定位时加速。
MySQL 不会直接“按行”从磁盘读取数据;它以 page(页)为单位加载数据,InnoDB 默认页大小是 16KB。一次查询哪怕只查 1 行,只要该行不在缓冲池中,就得把整页(含可能几十行)从磁盘读入 innodb_buffer_pool。
buffer_pool:命中则直接返回,不碰磁盘extent,64 个连续页)预读,但实际仍以页为单位载入和遍历过滤发生在两层协同完成,但关键判断在存储引擎层:
WHERE 中能被索引覆盖的条件(如 id = 5、status IN (1,2)),由 InnoDB 在读取页后、构造记录前就完成快速跳过(使用索引 B+ 树导航 + 页内二分/游标遍历)UPPER(name) = 'ABC'、涉及函数或跨列计算),会在 Server 层对引擎返回的每行做二次计算过滤EXPLAIN 显示 type=ref,若 Extra 列出现 Using where,说明仍有 Server 层过滤——这意味引擎返回了多余行,增加了网络和 CPU 开销有本质区别,尤其当表包含大字段(TEXT、BLOB)或行长度较大时:
SELECT *:InnoDB 必须读取整行物理记录(包括溢出页中的大字段),即使你最终不使用它们SELECT col1, col2(且这两列都在主键页内):InnoDB 可启用 read view + record read 优化,只读取所需列对应的数据,跳过溢出页(前提是没用到 SELECT * 或隐式需要全部列的场景,如触发 TRIGGER)Using index),那无论写 * 还是具体列,都只读索引页,不回表——此时列数不影响 I/O,只影响网络传输量因为 LIMIT 是结果截断,不是执行剪枝:
ORDER BY 配合 LIMIT(如 ORDER BY created_at DESC LIMIT 1),而 created_at 没索引,就要排序全部数据再取头,LIMIT 完全无效WHERE status=1 ORDER BY id ASC LIMIT 1,且 (status, id) 有联合索引)EXPLAIN 的 rows 值是否显著下降;若仍是全表行数,LIMIT 就没省 I/OEXPLAIN SELECT * FROM orders WHERE user_id = 123 ORDER BY created_at DESC LIMIT 1;
真正影响读取效率的,从来不是 SQL 写法表面的简洁,而是数据如何组织、索引如何设计、以及每一行背后的页加载与过滤时机。很多“慢查询”优化,本质是把 “让 MySQL 少读几个页” 变
