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.这就是一个我不了解的知识点.按照补全的写,充分了解每一列可以插入怎样的值.

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,992评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,212评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,535评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,197评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,310评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,383评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,409评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,191评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,621评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,910评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,084评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,763评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,403评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,083评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,318评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,946评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,967评论 2 351

推荐阅读更多精彩内容

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