Mysql --思考NULL

在创建表时,通过是否指定NOT NULL,可以指定列是否可以不包含值.当一个列不包含值时,称其为包含值NULL.NULL与字段包含0,空字符串或仅仅包含空格不同.下面开始对NULL进行讲解.

NOT NULL就一定不可以插入NULL吗?

答案是否定的.创建一下三个表进行验证.

CREATE TABLE `test_null_1` (
  `id` mediumint(8)  NOT NULL AUTO_INCREMENT,
  `create_time` timestamp NOT NULL ,
   PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8
CREATE TABLE `test_null_2` (
  `id` mediumint(8)   AUTO_INCREMENT,
  `create_time` timestamp  ,
   PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8
CREATE TABLE `test_null_3` (
  `id` mediumint(8) not null  AUTO_INCREMENT,
 `gender` int not null,
  `name` char(22) not null,
`create_time` datetime not null,
   PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

在三个表中插入数据

insert into test_null_1 values(null,null);
insert into test_null_2 values(null,null);
insert into test_null_3(gender) values(null);
insert into test_null_3(name) values(null);
insert into test_null_3(create_time) values(null);

插入结果如图所示:


从以上插入结果中可以看出,如果列定义了NOT NULL属性,则是不允许插入NULL值的,若没有规定则可以插入NULL值.但是有两个特例:

  1. 具有auto_increment属性的列,即使此列规定NOT NULL,仍然可以插入Null值,且插入成功
  2. 时间类型为 Timestamp的列,即使此列规定NOT NULL,仍然可以插入Null值,且插入成功.(注意:并不是所有的时间类型都是特例,在test_null_3中验证了datetime类型如果规定NOT NULL,则不可以插入NULL值,date,time和datetime一样,都不是特例)

NULL不一定为NULL

插入成功的NULL在表中存储的就一定是NULL吗?
答案是否定的.
创建表test_null_4

CREATE TABLE `test_null_4` (
  `id` mediumint(8) not null  AUTO_INCREMENT,
 `gender` int ,
  `name` char(22) ,
 `create_time` datetime ,
   PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

插入数据

insert into test_null_4(gender,name,create_time)values(null,null,null)

查询test_null_1,test_null_2,test_null_4表中数据,结果如图所示


除了两个特例,其他数据类型的列,如果允许插入NULL,且插入NULL的话,则在表中存储就是NULL.即NULL就是NULL
对于两个特例而言,插入NULL,但在表中显示的结果就不是NULL了:

  1. 具有auto_increment属性的列,即使此列规定NOT NULL,如果往这属性的列中插入Null值的话,仍可以插入NULL成功,且系统会插入一个正整数序列。
  2. 时间类型为 Timestamp的列,即使此列规定NOT NULL,如果往这属性的列中插入Null值的话,仍可以插入NULL成功,且系统会插入一个当前时间.
    为什么系统会默认插入一个当前时间呢?用show命令查询创建表的语句
#自己定义
CREATE TABLE `test_null_1` (
  `id` mediumint(8)  NOT NULL AUTO_INCREMENT,
  `create_time` timestamp NOT NULL ,
   PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8
#系统补全
 CREATE TABLE `test_null_1` (
  `id` mediumint(8) NOT NULL AUTO_INCREMENT,
  `create_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB  DEFAULT CHARSET=utf8
#自己定义
CREATE TABLE `test_null_2` (
  `id` mediumint(8)   AUTO_INCREMENT,
  `create_time` timestamp  ,
   PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8
#系统补全
 CREATE TABLE `test_null_2` (
  `id` mediumint(8) NOT NULL AUTO_INCREMENT,
  `create_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 

从中可以看出,对于timestamp 类型,即使我们没有规定NOTNULL属性,系统也会帮我们补全,并设置一个当前时间的默认值.

列中到底插入了什么值?

在生成记录却没有插入此列值时,那么在这条记录中此列最终会插入什么值呢?
首先毫无疑问,具有auto_increment属性的列插入的永远都是以正整数.那么其他列呢?
插入一下数据(我删掉了之前插入的数据)

insert into test_null_1 (id)values(null);
insert into test_null_2 (id)values(null);
insert into test_null_3 (id)values(null);
insert into test_null_4 (id)values(null);

对于以上结果,在来看看系统帮我们自动补全了什么

#自己定义
CREATE TABLE `test_null_3` (
  `id` mediumint(8) not null  AUTO_INCREMENT,
 `gender` int not null,
  `name` char(22) not null,
`create_time` datetime not null,
   PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
#系统补全
CREATE TABLE `test_null_3` (
  `id` mediumint(8) NOT NULL AUTO_INCREMENT,
  `gender` int(11) NOT NULL,
  `name` char(22) NOT NULL,
  `create_time` datetime NOT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8 |
#自己定义
CREATE TABLE `test_null_4` (
  `id` mediumint(8) not null  AUTO_INCREMENT,
 `gender` int ,
  `name` char(22) ,
 `create_time` datetime ,
   PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
#系统补全
CREATE TABLE `test_null_4` (
  `id` mediumint(8) NOT NULL AUTO_INCREMENT,
  `gender` int(11) DEFAULT NULL,
  `name` char(22) DEFAULT NULL,
  `create_time` datetime DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=5 DEFAULT CHARSET=utf8

从以上补全内容可以看出,当列没有规定NOT NULL也没有设置默认值时,系统会帮我们自动补全一个NULL默认值.也就是说,当列没有规定NOT NULL时,不管用户显示定义,还是系统自动补全,列都应该有且会有一个默认值.
当规定NOT NULL,用户也可以设置默认值,当用户没有设置时,系统默认值会根据不同数据类型而定,对于int,,会默认插入0;对于char,会默认插入空字符串;对于datetime,date,time三种时间数据类型,会默认插入零点, 0000-00-00 00:00:00
ps:从以上可以看出,对于四种时间类型,datetime,date,time,timestamp,
timestamp类型是多么与众不同.不管是否允许插入空值,都会默认插入当前时间.

NULL值检查

NULL 的条件比较运算比较特殊的,不能使用 = NULL 或 != NULL 在列中查找 NULL 值 。需要使用IS NULL 和 IS NOT NULL 运算符。

聚集函数对于NULL的处理

SQL聚集函数,我们经常用到的有AVG(),COUNT(),MAX(),MIN(),SUM(),这几个函数都是忽略值为NULL的行的.

1.count()
CREATE TABLE `test_null_5` (
  `id` mediumint(8)  NOT NULL AUTO_INCREMENT,
  `age` int(11) default NULL,
  `name` char(22) default NULL,
  `create_time` timestamp NOT NULL  DEFAULT CURRENT_TIMESTAMP,
   PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
insert into test_null_5 values(null, null,null,null);
insert into test_null_5 values(2, 20,'sunyan',now());
insert into test_null_5 values(null, 22,'',now());
insert into test_null_5 values(null, 0,'',now());
select * from test_null;
select count(age),count(name), count(*) from test_null;

从查询结果中可以看到,count()函数有两种使用方式,这两种方式对于NULL的处理略有不同.

  1. count(*)是对表中 行的数目进行计数,女给不管表列中包含的是空值NULL还是非空值.
  2. count(column)对特定列中具有值的行进行计数,忽略NULL值.
2.AVG()

从结果中可以看出,AVG()忽略值为NULL的行

反思一下

  1. 最近因为NULL值,意识到平时一些习以为常的知识点,其实我并不能了解他.
  2. 以后在写创建表SQL时,一定要按照系统会补全的样子写.像我以前可能会这样定义一张表.
CREATE TABLE `test_null_4` (
  `id` mediumint(8) not null  AUTO_INCREMENT,
 `gender` int ,
  `name` char(22) ,
 `create_time` datetime ,
   PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

对比系统补全语句,我发现自己在这样定义时,完全没有意识到gender会有一个默认值NULL.这就是一个我不了解的知识点.按照补全的写,充分了解每一列可以插入怎样的值.

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,292评论 19 139
  • SQL SELECT 语句 一、查询SQL SELECT 语法 (1)SELECT 列名称 FROM 表名称 (2...
    有钱且幸福阅读 11,051评论 0 33
  • mysql(5.5)所支持的日期时间类型有:DATETIME、 TIMESTAMP、DATE、TIME、YEAR。...
    snoweek阅读 5,673评论 0 0
  • MySQL技术内幕:SQL编程 姜承尧 第1章 SQL编程 >> B是由MySQL创始人之一Monty分支的一个版...
    沉默剑士阅读 7,194评论 0 3
  • 我是一只猫,一只花白猫,一只懒惰的花白猫,一只丑陋又懒惰的花白猫。 现在我正躺在领居家的屋顶上,我已经快要没有一点...
    一苇以航tmy阅读 1,644评论 0 0