浅谈 NULL 和 空值的区别
NULL也就是在字段中存储NULL值
空字符串值也就是字段中存储空字符('')
- 空值''是不占用空间的,mysql中的NULL其实是占用空间的
- NULL 其实并不是空值,而是要占用空间,所以mysql在进行比较的时候,所以对效率有一部分影响,而且B树索引时不会存储NULL值的,所以如果索引的字段可以为NULL,索引的效率会下降很多。
- 空值(null)是不能参与任何计算,因为空值参与任何计算都为空
- 在统计过程中,这个count函数会自动忽略空值的数据。此时统计出来的就是有值的信息。 如果采用的是空字符''的数据,则这个函数会将其统计进去
- 对于可为空的具有默认值的属性列 AUTO_INCREMENT、timestamp,会自动添加默认值
- 空值跟null的区别。mysql官方:
“NULL columns require additional space in the row to record whether their values are NULL. For MyISAM tables, each NULL column takes one bit extra, rounded up to the nearest byte.”
译文:“空列需要行中的额外空间来记录它们的值是否为NULL。对于MyISAM表,每个空列额外占用一位,四舍五入到最近的字节。”
我们来通过测试来看看 他们彼此的区别:
1. 占用空间区别
mysql> select length(NULL), length(''), length('1');
+--------------+------------+-------------+
| length(NULL) | length('') | length('1') |
+--------------+------------+-------------+
| NULL | 0 | 1 |
+--------------+------------+-------------+
1 row in set (0.03 sec)
小结 : 从上面的测试可以看出 字符串空值(’’)的长度是0,是不占用空间的, 而的NULL长度是NULL,其实它是占用空间的!
2、插入方式区别
#创建一个表,tb_test
create table tb_test(
id int unsigned primary key auto_increment,
one varchar(10) NOT NULL,
two varchar(255) DEFAULT NULL
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
插入进行验证:
#全部插入 NULL,会失败 原因就是指定的不允许插入NULL
insert into tb_test(one,two) value (NULL,NULL);
1048 - Column 'one' cannot be null
#全部插入 空字符串值,成功 原因就是 ('') 字符 和 NULL的类型都不一样 指定的是不允许插入NULL,又没有说不允许('')空字符串!^.^
insert into tb_test(one,two) value ('','');
Query OK, 1 row affected
#这也是刚刚讲过not null约束测试insert语句的时候, 插入('')空字符串会成功的原因!
3、在查询方式上的区别对比
#创建一个表,tb_test2
create table tb_test2(
id int unsigned primary key auto_increment,
one varchar(10) NOT NULL,
two varchar(255) DEFAULT NULL
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
#模拟数据:
insert into tb_test2(one,two) values (1,NULL);
insert into tb_test2(one,two) values ('',2);
insert into tb_test2(one,two) values (3,3);
#查询one字段
#使用 is null 来查询one字段
select * FROM tb_test2 where one is null; #结果就是一条也没有,因为one字段并没有代表为NULL的数据存在!
#使用 is not null 来查询one字段
select * FROM tb_test2 where one is not null; #结果被全部查询出来,因为one字段中的三个数据都不为NULL这个类型
#使用 = 和 != 来查询one字段
select * FROM tb_test2 where one ='';
select * FROM tb_test2 where one != '';
#查询two字段
#使用 is null 来查询two字段
select * FROM tb_test2 where two is null; #结果有一条符合NULL,
#使用 is not null 来查询two字段
select * FROM tb_test2 where two is not null; #结果是不符合NULL的有两条
#使用 = 来查询two字段
select * FROM tb_test2 where two ='';
#使用 != 来查询two字段
#这里要注意的是为NULL的并没有查询出来,原因用 != 来查 字符串空('')的时候, 会把NULL也当做是字符串空来判断吧!
select * FROM tb_test2 where two != '';
小结: 如果要单纯查NULL值列,则使用 is NULL去查,单纯去查空值(’’)列,则使用 =''。
建议查询方式:NULL值查询使用is null/is not null查询,而空值(’’)可以使用=或者!=、<、>等算术运算符来查!
4、在count()统计函数上的区别
#创建一个表,tb_test3
create table tb_test3(
id int unsigned primary key auto_increment,
one varchar(10) NOT NULL,
two varchar(255) DEFAULT NULL
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
#模拟数据:
insert into tb_test3(one,two) values (1,NULL);
insert into tb_test3(one,two) values ('',2);
insert into tb_test3(one,two) values (3,3);
#使用COUNT函数统计one字段:
select count(one) from tb_test3; #结果为: 3 条, 说明 空字符串('') 会被count()函数统计!
#使用COUNT函数统计two字段:
select count(two) from tb_test3; #结果为: 2条, 原因是NULL 不会被count()函数统计到!
#注意: 使用 * 号来统计会把NULL算进去!
SELECT count(*) FROM tb_test;
+----------+
| count(*) |
+----------+
| 3 |
+----------+
实际开发到底是使用NULL值还是空值(’’)呢?
根据实际业务来进行区分,个人建议在实际开发中如果没有特殊的业务场景,可以直接使用空字符串值(’’) !
mysql不用null的理由
(1) 所有使用NULL值的情况,都可以通过一个有意义的值的表示,这样有利于代码的可读性和可维护性,并能从约束上增强业务数据的规范性。
(2) NULL值到非NULL的更新无法做到原地更新,更容易发生索引分裂,从而影响性能。(null -> not null性能提升很小,除非确定它带来了问题,否则不要当成优先的优化措施)
(3) NULL值在timestamp类型下容易出问题,特别是没有启用参数explicit_defaults_for_timestamp
(4) NOT IN、!= 等负向条件查询在有 NULL 值的情况下返回永远为空结果,查询容易出错
(5) 列值允许为空,索引不存储null值,结果集中不会包含这些记录。
(6) 使用concat拼接时,首先要对各个字段进行非null判断,否则只要任何一个字段为空都会造成拼接的结果为null
(7) 当计算count时候null column不会计入统计
(8) 存在空值的列,加上索引后,还是会走索引的,但是索引里面不包含空值
(9) 针对非空列的查询的时候会走全表扫描
...