SQL optimizer_switch 的 index_merge_union 开关效果实测

14次阅读

mysql 的 index_merge_union 需同时开启 index_merge 和 index_merge_union,且仅在 or 连接的独立单列等值条件、各字段有高基数单列索引时才触发;性能通常不如联合索引,应优先建合适联合索引而非依赖该优化。

SQL optimizer_switch 的 index_merge_union 开关效果实测

index_merge_union 开启后为啥没走索引合并?

MySQL 的 index_merge_union 不是“开了就自动生效”的开关,它只在优化器判定多个单列索引的并集比全表扫描或单索引更优时才触发。常见现象是:明明开了,EXPLAIN 里却看不到 type=unionExtra=Using union(……)

实操建议:

  • 必须确保 WHERE 条件是 OR 连接的、彼此独立的单列等值条件(如 a = 1 OR b = 2),不能含函数、范围查询(>BETWEEN)、IS NULL 等;
  • 每个涉及字段都要有独立的单列索引(不是联合索引);
  • SELECT …… FROM t WHERE a = 1 OR b = 2 这类简单语句测试,避免 JOIN 或子查询干扰优化器决策;
  • 检查 SHOW INDEX FROM t,确认 ab 索引的 Cardinality 都足够高(低基数字段如状态码通常不触发)。

optimizer_switch 中 index_merge_union 的位置和依赖

index_merge_unionoptimizer_switch 的一个子项,但它依赖于 index_merge 总开关——如果 index_merge=off,单独开 index_merge_union 无效。

实操建议:

  • 查看当前设置:执行 SELECT @@optimizer_switch,搜索是否同时包含 index_merge=onindex_merge_union=on
  • 安全开启方式:SET SESSION optimizer_switch='index_merge=on,index_merge_union=on';(会话级,不影响其他连接);
  • 不要只写 index_merge_union=on,漏掉 index_merge=on 是最常踩的配置坑;
  • 全局修改需谨慎:SET GLOBAL optimizer_switch=…… 会重置所有未显式指定的子项为默认值,建议完整拼出整个字符串。

union 索引合并 vs 联合索引,哪个更快?

即使 index_merge_union 触发成功,性能也不一定比建一个合适的联合索引好。MySQL 的索引合并要分别扫描多个索引、去重、归并结果集,IO 和 CPU 开销都更高。

实操建议:

  • 优先考虑建联合索引:对 WHERE a = ? OR b = ?INDEX(a), INDEX(b) 可能不如 INDEX(a,b)INDEX(b,a)(取决于查询频率和选择性);
  • SELECT COUNT(*) 对比耗时:分别在开启 / 关闭 index_merge_union 下执行相同 OR 查询,看执行时间与 Handler_read_* 状态变量变化;
  • 注意数据分布:当两个条件返回行数差异极大(比如一个返回 10 行,另一个返回 10 万行),合并代价可能远超单索引 + OR 逻辑改写(如 UNION DISTINCT);
  • 5.7+ 版本中,index_merge_intersectionAND 场景下更稳定,但 union 类型始终是优化器最后考虑的备选方案。

EXPLAIN 看不到 Using union?检查这几个地方

EXPLAIN 输出里没出现 Using union(……),不代表没走索引合并——也可能是 MySQL 用了 index_merge_intersection 或降级为 range 扫描。

实操建议:

  • 务必加 FORMAT=JSON:执行 EXPLAIN FORMAT=JSON SELECT ……,在 query_block -> access_typeusing_join_buffer 上下文里找 union 字样;
  • 开启 optimizer trace:SET optimizer_trace="enabled=on"; 再执行查询,查 information_schema.OPTIMIZER_TRACE,里面会明确写出“considering union access”或“rejected by cost calculation”;
  • 留意 type 字段:如果是 index_merge,且 key 显示多个索引名(如 key: a,b),基本可确认触发;
  • 避免隐式类型转换:比如 WHERE varchar_col = 123 会导致索引失效,合并自然也不会发生。

真正难的是判断“该不该开”——它解决的是没有联合索引又不得不写 OR 的临时场景,而不是索引设计的替代方案。线上环境一旦发现 index_merge_union 频繁生效,第一反应应该是补索引,而不是调优开关。

text=ZqhQzanResources