为什么不建议给MySQL设置Null值?

之前ElasticSearch系列文章中提到了如何处理空值,若为Null则会直接报错,因为在ElasticSearch中当字段值为null时、空数组、null值数组时,会将其视为该字段没有值,最终还是需要使用exists或者null_value来处理空值
大多数ElasticSearch的数据都来自于各类数据库,这里暂且只针对于MySQL,各个开源软件中都默认兼容各种Null值,空数组等等
若从根源上截断就可以省很多事,直到现在很多开发小伙伴还是坚韧不拔的给字段的默认值还是Null
本期就来聊一聊为什么不建议给字段的默认值设置为Null
本期环境为:MySQL8.0.26

为什么不建议给MySQL设置Null值?

文章插图
 
null
一、案例数据创建表user
CREATE TABLE `user` ( `id` int(11) unsigned NOT NULL AUTO_INCREMENT, `name` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci DEFAULT NULL, `age` tinyint(4) unsigned NOT NULL, PRIMARY KEY (`id`)) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci添加数据,共计10条数据,有两条数据的name值为Null
INSERT INTO `user` (`name`, `age`) VALUES ('kaka', 26);INSERT INTO `user` (`name`, `age`) VALUES ('niuniu', 26);INSERT INTO `user` (`name`, `age`) VALUES ('yangyang', 26);INSERT INTO `user` (`name`, `age`) VALUES ('dandan', 26);INSERT INTO `user` (`name`, `age`) VALUES ('liuliu', 26);INSERT INTO `user` (`name`, `age`) VALUES ('yanyan', 26);INSERT INTO `user` (`name`, `age`) VALUES ('leilie', 26);INSERT INTO `user` (`name`, `age`) VALUES ('yao', 26);INSERT INTO `user` (`name`, `age`) VALUES (NULL, 26);INSERT INTO `user` (`name`, `age`) VALUES (NULL, 26);一、count数据丢失在这期 MySQL统计总数就用count,别花里胡哨的《死磕MySQL系列 十》 文章中,已经对count的使用说的非常明白了 。
那借着这个案例,来分析一下为什么数据会丢失,先看结果
select count(*) as num1 ,count(name) as num2 from user;
为什么不建议给MySQL设置Null值?

文章插图
 
使用count字段名时出现了数据丢失,很明显是因为主键ID9、10这两条记录的name值为空造成的 。
为什么会出现这种情况?
当count除了主键字段外,会有两种情况:
一种是字段为null,执行时,判断到有可能是null,但还要把值取出来再判断下,不是null的进行累加
另一种是字段为not null,执行时,逐行从记录里边读出这个字段,判断不是null,才进行累加
此时,咱们遇到的问题是name字段的值存在了null值,所以会走第一种情况,不进行统计null值
为什么建议大家都使用count(*)?
MySQL对于count做了专门的优化,跟字段不同的是并不是把所有带了*的值取出来,而是指定了count(*)肯定不是null,只需要按行累加即可
MySQL团队对count(*)做了什么优化?
MySQL系列文章至今已经更新了第十八期了,你有没有猜到原因呢?
现在你应该知道主键索引结构中叶子节点存储的是整行数据,而普通索引叶子节点存储的是主键ID
那对于普通索引来说肯定会比主键索引小,因为对于MySQL来说,不管遍历哪个索引结果都一样,所以优化器会主动去找到那颗最小的树进行遍历 。
在逻辑正确的前提下,尽量减少访问数据量,是数据库系统设计通用法则之一 。
【为什么不建议给MySQL设置Null值?】最后给大家留一个问题,为什么Innodb存储引擎不跟Myisam存储一样存储一个count值呢?
如果不知道的话,可以看上文提到的count文章
二、为distinct打抱不平在开发工作中使用Distinct进行去重的场景十分的少,大多数情况都是使用group by完成的
select distinct name from user;可以看到此时的数据依然是正确的,对Null值做了去重的操作
为什么不建议给MySQL设置Null值?

文章插图
 
为什么要说这个,因为咔咔在其它的平台上看到过有人这么使用count(distinct name,mobile),然后说是统计出来的数据不准确 。
这种用法依然是count(字段)的用法,distinct本身是会对Null进行去重,去重后依然是需要判断name的值不为null时,才会进行累计 。
所以,不要把锅甩给distinct
三、使用表达式数据丢失在一些值为null时,使用表达式会造成数据的不一致,接下来一起看下
select * from user where name != 'kaka';
为什么不建议给MySQL设置Null值?

文章插图
 
表达式造成数据丢失


推荐阅读