Skip to main content

15.IS NULL 优化

MySQL 可以对 col_name IS NULL 执行与 col_name = Constant_value 相同的优化。 例如,MySQL 可以使用索引和范围通过 IS NULL 搜索 NULL。

示例:

SELECT * FROM tbl_name WHERE key_col IS NULL;

SELECT * FROM tbl_name WHERE key_col <=> NULL;

SELECT * FROM tbl_name
WHERE key_col=const1 OR key_col=const2 OR key_col IS NULL;

如果 WHERE 子句包含声明为 NOT NULL 的列的 col_name IS NULL 条件,则该表达式将被优化掉。 如果列无论如何都可能生成 NULL(例如,如果它来自 LEFT JOIN 右侧的表),则不会发生此优化。

MySQL 还可以优化 col_name = expr OR col_name IS NULL 的组合,这是解析子查询中常见的形式。 当使用此优化时,EXPLAIN 显示 ref_or_null。

此优化可以处理任何关键部分的 IS NULL。

一些优化查询的示例,假设表 t2 的 a 列和 b 列上有索引:

SELECT * FROM t1 WHERE t1.a=expr OR t1.a IS NULL;

SELECT * FROM t1, t2 WHERE t1.a=t2.a OR t2.a IS NULL;

SELECT * FROM t1, t2
WHERE (t1.a=t2.a OR t2.a IS NULL) AND t2.b=t1.b;

SELECT * FROM t1, t2
WHERE t1.a=t2.a AND (t2.b=t1.b OR t2.b IS NULL);

SELECT * FROM t1, t2
WHERE (t1.a=t2.a AND t2.a IS NULL AND ...)
OR (t1.a=t2.a AND t2.a IS NULL AND ...);

ref_or_null 的工作原理是首先读取引用键,然后单独搜索具有 NULL 键值的行。

优化只能处理一个 IS NULL 级别。 在以下查询中,MySQL 仅对表达式 (t1.a=t2.a AND t2.a IS NULL) 使用键查找,并且无法使用 b 上的键部分:

SELECT * FROM t1, t2
WHERE (t1.a=t2.a AND t2.a IS NULL)
OR (t1.b=t2.b AND t2.b IS NULL);