由null出发-看看为何mysql这么排斥null

前言

在表结构设计时我们经?;崦娑允欠褚涌罩翟际⒛现荡淼任侍猓毕蚯叭司橹屑焖魇?,经?;峥吹讲唤ㄒ槲猲ull,强刷面经时not null也经常被归结到优化建议当中去,今天就来看一下为何?
通常来说,null表示虚无&不确定,在不同的实现中对null的定义相似。最早在Codd提出关系模型的最早的paper里,就引入了NULL。
先看不同语言对于null的处理:

c++的NULL

c++中NULL是一个宏,是一个空指针常量,如果将NULL扩展为常数,那么这个数是0,类型为int,也就是说常数0既是整型常量,也是空指针常量(cpp 11中引入了nullptr作为空指针常量,解决二义性)。

Java的null

Java中的null是一个关键字,通常指的是非基础类型变量的引用指向一个“空”,可以理解为引用对象为不确定,Java是一种基于copy传递的基础类型+引用类型数据构成的对象体系,Java内存管理中使用句柄来操控对象,而null表示句柄所在的内存地址上没有存储任何真实对象的地址。

go的nil

go中没有null这个关键字(任何类型都会零值处理),但是有一个类似的nil(注意不是关键字),跟null的定位类似但不是完全相同,是一个预先定义的标识符,通常用于指针类型、map、slice、function、channel、interface等类型的零值定义,是一个没有默认类型的空值,标示的并非不确定性,而是可以明确代表各种不同内存布局的类型的零值。
比如说:

fmt.Println( (interface{})(nil) == (*int)(nil) ) 
// print:false

//下面这样声明零值后是 零值间是无法比较的
var _ = ([]int)(nil) == ([]int)(nil)
var _ = (map[string]int)(nil) == (map[string]int)(nil)
var _ = (func())(nil) == (func())(nil)

然后来看存储中对null的定义:

redis中的空值

redis中操作时都是针对明确的值的,不会出现像是不确定这样的定义,通常来说,所以在操作redis时是没有对null的数据定义的,但是会有nil这样的零值存在,跟go处理方式比较像,代表n个结构的零值,一些语言对于nil return这种情况有时候会变成一种null来处理,因为对于client来说,这就是个不确定的返回或者空值。

127.0.0.1:6379> get B
(nil)

说到redis了,也顺道看下lua对于空返回的处理,容易写出bug来,简单记录了下,与本章主题无关。官方文档中的:
A non-nil Redis "bulk reply" results in a Lua string as the return value.
A nil bulk reply results in a ngx.null return value

mysql的空值定义

看到各种语言、工具中对null、nil等关键词的使用,不难发现 null代表的是一种内存存储的不确定性,nil代表的通常是一种约定的零值。
mysql中也是如此,null代表的是一种不确定性,所以通常用is null 或者not null来判定一个实例数据是否为不确定的,而不是直接==来进行值比较。
官方手册中中的定义:
The NULL value means “no data.” NULL can be written in any lettercase
也就产生了下面的关系操作符:
IS NULL\IS NOT NULL:能正常比较
<=>:结果都是false
mysql 这样对null的定义导致了一些问题:
1、误导性&sql操作时的疏忽

select A from t_xx where B not null or B = 2\G #正常操作
select A from t_xx where B = null or B = 2\G # B = null条件异常
select * from tableA where xxx not in(1, 2, null)\G #返回永远为空
select count(*) from tableA\G #null不参与统计

2、另外,null值在timestamp类型下容易出问题,特别是没有启用参数explicit_defaults_for_timestamp

mysql null的空间表现

null在mysql中是一种特殊的“占位符”,用来表示不确定性,但是实际上它也是需要占用一部分内存空间的,比不是所想的会省内存。
null的长度是null,Compact Row Format前提下,每个行记录都会有一个Bit vector来记录行中出现NULL的字段,长度为 N / 8 向上取整,其中 N为值NULL的字段数。

select length(''), length(null), length(0), length('0');
 +------------+--------------+-----------+-------------+
 | length('') | length(null) | length(0) | length('0') |
 +------------+--------------+-----------+-------------+
 |          0 |         NULL |         1 |           1 |
 +------------+--------------+-----------+-------------+

对于变长字段是可以起到一定的省空间的作用,对于int/char这些是毫无作用的,并且需要一个额外字节作为判断是否为null的标志位(这个是问题,但忽略不计)
某些情况可以省,但是相对于它带来的其他影响,这点优势啥都不是,继续往下看

mysql null对索引的影响

Mysql难以优化引用可空列查询,它会使索引、索引统计和值更加复杂??煽樟行枰嗟拇娲⒖占?,还需要mysql内部进行特殊处理??煽樟斜凰饕?,每条记录都需要一个额外的字节,还能导致MYisam 中固定大小的索引变成可变大小的索引。 --《高性能mysql》
1、虽然mysql会对null字段也进行索引,但是只有is null的方式会使用上索引,所以一旦使用不好,索引就无效了。
2、如果null列做了唯一索引,那就尴尬了:唯一索引字段允许插入多条null的记录
3、null一定程度上会是key_len变长(key_len 的计算规则和三个因素有关:数据类型、字符编码、是否为null,上面也提到啦)
4、mysql内部会对null 做很多特殊逻辑的处理影响性能。

总结

1、使用上容易出错
2、很容易导致索引不可用或者效率下降
3、很多情况下会增加存储
4、提升表的维护成本
所有使用NULL值的情况,都可以通过一个有意义的值的表示,这样有利于代码的可读性和可维护性。

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