解读索引列中有null值会不会使索引失效

 更新时间:2023年12月13日 14:27:44   作者:zyjzyjjyzjyz  
这篇文章主要介绍了解读索引列中有null值会不会使索引失效问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教

先说答案

null不会使索引失效,但是会影响优化器对执行计划的选择。

网上很多都说null会导致索引失效,这么说并不严谨。先看实验。

注意:

  • count(列)不会把空值算进去。
  • distance 列 如果列中有null会把列当成一行输出。
  • count(*)会把null值算进去。

实验1

create table null_test(
 id int PRIMARY KEY,
 name VARCHAR(10),
 age VARCHAR(10),
 KEY inx_test_age(age),
 KEY inx_test_name(name)
)
 
insert into null_test values(1,'a','2');
insert into null_test values(2,'b','3');
insert into null_test values(3,'c','4');
insert into null_test values(4,'d','5');
insert into null_test values(5,null,'6');
insert into null_test values(6,null,'6');
insert into null_test values(7,null,'9');
insert into null_test values(8,'q',null);
insert into null_test values(9,'','5');
insert into null_test values(10,'','7');
insert into null_test values(11,'t','');

创建null_test表,并在name、age列上建普通索引,插入null值。

explain
select * from null_test where name is null;

可以看到name  is  null走了索引,并且type是ref,这是普通索引的等职查询才会有的。

对于explain的详解:explain性能详细分析

explain 
select * from null_test where name is not null;

可以看到name  is  not  null确实没有走索引,而是全表扫描。这意味着导致索引失效吗?往下看。

实验2

create table null_test2(
 id int PRIMARY KEY,
 name VARCHAR(10),
 age VARCHAR(10),
 KEY inx_test2_age(age),
 KEY inx_test2_name(name)
)
 
 
insert into null_test2 values(1,'a','2');
insert into null_test2 values(2,'b','3');
insert into null_test2 values(3,'c','4');
insert into null_test2 values(4,'d','5');
insert into null_test2 values(5,null,'6');
insert into null_test2 values(6,null,'6');
insert into null_test2 values(7,null,'9');
insert into null_test2 values(8,null,'6');
insert into null_test2 values(9,null,'6');
insert into null_test2 values(10,null,'9');
insert into null_test2 values(11,null,'9');
insert into null_test2 values(12,null,'6');
insert into null_test2 values(13,null,'6');
insert into null_test2 values(14,null,'9');

创建null_test2表,插入很多null值。

explain
select * from null_test2 where name is null;

可以看到和上面的条件都是相同的,但是却是走了全表扫描,还没想明白?接着往下看。

explain 
select * from null_test2 where name is not null;

可以看到name  is  not  null走了索引,和上面的情况正好相反,这是什么情况?

  • 其实这和普通索引上的情况相同,我们把null值当成正常的值,mysql默认认为null是相同的,所以重复率特别高的话,优化器肯定不会走索引,而是走全表扫描。
  • 还要注意一点,is null时type=ref,is  not  null时type=range。

实验3

create table null_test3(
 id int PRIMARY KEY,
 name VARCHAR(10),
 age VARCHAR(10),
 KEY inx_test2_age(age),
 UNIQUE KEY inx_test2_name(name)
)
 
insert into null_test3 values(1,'a','2');
insert into null_test3 values(2,'b','3');
insert into null_test3 values(3,'c','4');
insert into null_test3 values(4,'d','5');
insert into null_test3 values(5,null,'6');
insert into null_test3 values(6,null,'6');
insert into null_test3 values(7,null,'9');
insert into null_test3 values(8,null,'6');
insert into null_test3 values(9,null,'6');
insert into null_test3 values(12,'q',null);
insert into null_test3 values(13,'','5');
insert into null_test3 values(10,'g','7');
insert into null_test3 values(11,'t','');
explain
select * from null_test3 where name is null;

explain
select NAME from null_test3 where name is null;

可以看到唯一索引也可以插入多个null,并且null就在索引上,因为使用索引就可以查到。

总结

上面我说过mysql内部认为null是相等的,所以导致当插入过多null值,造成重复率过多,is null不会走索引。而is  not  null因为查询的结果过多,优化器选择了全表扫描。

什么原因让mysql认为null是相等的:

其实是有个参数控制的。

innodb_stats_method

show variables like 'innodb_stats_method';
SET GLOBAL  innodb_stats_method=nulls_unequal;

该参数有三个值,默认为nulls_equal

1、null_equal:认为所有的null值都是相等的,也是默认值,这种统计方式,会让优化器认为某个列中的平均一个值的重复次数特别多,倾向于不适用索引去访问。

2、nulls_unequal:认为所有的null值都不相等,这种统计方式,会让优化器认为某个列中的平均一个值的重复次数特别少,更倾向于使用索引去访问。

3、nulls_ignored:直接忽略null

在mysql5.7.2版本之后,mysql将这个值写死为nulls_equal

好了,以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • MySQL中的空格处理方法

    MySQL中的空格处理方法

    在MySQL中,空格是一个特殊的字符,本文主要介绍了MySQL中的空格处理方法,具有一定的参考价值,感兴趣的可以了解一下
    2023-11-11
  • mysql limit分页优化方法分享

    mysql limit分页优化方法分享

    MySQL的优化是非常重要的。其他最常用也最需要优化的就是limit。MySQL的limit给分页带来了极大的方便,但数据量一大的时候,limit的性能就急剧下降。
    2011-04-04
  • 使用python连接mysql数据库之pymysql模块的使用

    使用python连接mysql数据库之pymysql模块的使用

    这篇文章主要介绍了使用python连接mysql数据库之pymysql模块的使用,本文给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下
    2019-09-09
  • 详解Mysql命令大全(推荐)

    详解Mysql命令大全(推荐)

    本篇文章详细的介绍了Mysql命令,MySQL是一个关系型数据库管理系统,由于其体积小、速度快、总体拥有成本低,尤其是开放源码这一特点,一般中小型网站的开发都选择MySQL作为网站数据库。
    2016-11-11
  • mysql一对多关联查询分页错误问题的解决方法

    mysql一对多关联查询分页错误问题的解决方法

    这篇文章主要介绍了mysql一对多关联查询分页错误问题的解决方法,非常不错,具有一定的参考借鉴价值,需要的朋友可以参考下
    2018-09-09
  • Windows Server 2019部署MySQL 8完整步骤教程

    Windows Server 2019部署MySQL 8完整步骤教程

    MySQL在开发开源软件时经常被当作该软件的数据管理系统,所以我们在开发时将会经常用到它,所以如何安装MySQL就是一个问题了,这篇文章主要介绍了Windows Server 2019部署MySQL 8的相关资料,需要的朋友可以参考下
    2026-04-04
  • MySQL 原理与优化之Limit 查询优化

    MySQL 原理与优化之Limit 查询优化

    这篇文章主要介绍了MySQL 原理与优化之Limit 查询优化,文章围绕主题展开详细的内容介绍,具有一定的参考价值,需要的小伙伴可以参考一下
    2022-08-08
  • MySQL首次登录跳过密码验证并修改密码实现方式

    MySQL首次登录跳过密码验证并修改密码实现方式

    文章介绍了如何在MySQL中重置忘记密码的步骤,包括查找和配置my.ini文件、跳过密码验证、连接数据库、修改密码以及恢复配置和重启服务
    2025-11-11
  • MySQL数据类型全解析

    MySQL数据类型全解析

    这篇文章主要介绍了MySQL数据类型的相关资料,帮助大家更好的理解和使用MySQL数据库,感兴趣的朋友可以了解下
    2021-01-01
  • MySQL修改root密码的4种方法(小结)

    MySQL修改root密码的4种方法(小结)

    这篇文章主要介绍了MySQL修改root密码的4种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-09-09

最新评论