IN和BETWEEN在索引效能的区别

1. 核心区别:等值遍历 vs. 连续扫描

IN:视为多个等值点(Point Search)

当你使用 IN (1, 2, 3) 时,MySQL 优化器通常会将其处理为:

  • 先去找 1 的位置,再去找 2 的位置,最后去找 3 的位置。
  • 关键点 :在联合索引 (a, b) 中,如果 a 使用了 IN,因为 a 的每一个值都是确定的,MySQL 可以继续利用索引的有序性去过滤 b
>, <, BETWEEN:视为线段扫描(Range Scan)

当你使用 a > 10 时,MySQL 认为 a 是一个连续的区间。

  • 关键点 :虽然 a 是有序的,但在 a > 10 这个大区间内,b无序 的(只有在 a 相等的情况下 b 才有序)。
  • 结果 :索引的效能在 a 字段就停止了,后续的 b 字段无法利用索引进行快速定位。

2. 实验对比

假设索引为 (age, score)

场景 A:使用 IN
sql 复制代码
SELECT * FROM users WHERE age IN (20, 30) AND score > 80;
  • MySQL 的动作
    1. 去索引树找 age=20 的位置,并在该范围内利用索引寻找 score > 80
    2. 再去找 age=30 的位置,并在该范围内利用索引寻找 score > 80
  • 结论agescore 两个字段都用到了索引
场景 B:使用 BETWEEN>
sql 复制代码
SELECT * FROM users WHERE age BETWEEN 20 AND 30 AND score > 80;
  • MySQL 的动作
    1. 找到 age=20 的起始位置,开始往后扫描,直到 age=30
    2. 在扫描过程中,它无法通过索引"跳跃"找到 score > 80 的记录,只能一条条检查 score 是否符合要求。
  • 结论只有 age 用到了索引,score 索引失效

3. 性能层面的区别

特性 IN (val1, val2) >, <, BETWEEN
底层分类 多个等值查询 (Multiple Equalities) 范围查询 (Range)
索引连续性 不截断后续索引字段 截断后续索引字段
排序 (Order By) 可能导致索引排序失效(需看具体执行计划) 必然导致后续字段排序失效
优化器开销 列表值越多,计算索引成本的开销越大 成本计算简单

4. 特殊情况:MySQL 8.0 的"跳跃扫描" (Skip Scan)

在 MySQL 8.0+ 中,如果联合索引的前缀列区分度很低(比如只有几个枚举值),即使你用了范围查询,优化器有时也会尝试通过"跳跃扫描"来强行利用后面的索引字段。但这种优化是不稳定的,最稳健的方法依然是把范围查询放在索引的最后


5. 什么时候 IN 也会变成"范围条件"?

虽然 IN 表现得像等值,但如果 IN 列表中的值过多 (超过了 eq_range_index_dive_limit 配置的值),或者它是通过一个复杂的子查询产生的,MySQL 可能会放弃点对点查找,将其降级为范围扫描。此时,它对后续索引列的影响就和 BETWEEN 一样了。

相关推荐
杉氧5 小时前
Navigation Compose 深度实践:如何优雅地串联起你的全栈 App?
android·架构·android jetpack
雨白9 小时前
指针与数组的核心机制
android
黄林晴13 小时前
Room 3.0 正式发布!包名彻底重构,KMP 成为核心主线
android·android jetpack
三少爷的鞋14 小时前
Kotlin 协程环境下的 DCL 懒加载:别把线程时代的经验直接搬过来
android
plainGeekDev14 小时前
Gson → kotlinx.serialization
android·java·kotlin
CYY951 天前
Compose 入门篇
android·kotlin
杉氧1 天前
Compose 时代的 MVI 架构:如何用单向数据流驱动复杂 UI?
android·架构·android jetpack
杉氧1 天前
Modifier 的艺术:为什么链式调用的顺序决定了UI 的生命周期?
android·架构·android jetpack
李斯维1 天前
腾讯 XLog 日志框架 Android 端接入
android·android studio·android jetpack