应用办公生活信息教育商业
快生活
商业财经
汽车智能
教育国际
房产环球
信息数码
热点科技
生活手机
晨报新闻
办公软件
科学动态
应用生物
体育时事

为什么数据库字段要使用NOTNULL?

5月13日 颜如初投稿
  最近刚入职新公司,发现数据库设计有点小问题,数据库字段很多没有NOTNULL,对于强迫症晚期患者来说,简直难以忍受,因此有了这篇文章。
  基于目前大部分的开发现状来说,我们都会把字段全部设置成NOTNULL并且给默认值的形式。
  通常,对于默认值一般这样设置:整形,我们一般使用0作为默认值。字符串,默认空字符串时间,可以默认1970010108:00:01,或者默认0000000000:00:00,但是连接参数要添加zeroDateTimeBehaviorconvertToNull,建议的话还是不要用这种默认的时间格式比较好
  但是,考虑下原因,为什么要设置成NOTNULL?
  来自高性能Mysql中有这样一段话:尽量避免NULL
  很多表都包含可为NULL(空值)的列,即使应用程序并不需要保存NULL也是如此,这是因为可为NULL是列的默认属性。通常情况下最好指定列为NOTNULL,除非真的需要存储NULL值。
  如果查询中包含可为NULL的列,对MySql来说更难优化,因为可为NULL的列使得索引、索引统计和值比较都更复杂。可为NULL的列会使用更多的存储空间,在MySql里也需要特殊处理。当可为NULL的列被索引时,每个索引记录需要一个额外的字节,在MyISAM里甚至还可能导致固定大小的索引(例如只有一个整数列的索引)变成可变大小的索引。
  通常把可为NULL的列改为NOTNULL带来的性能提升比较小,所以(调优时)没有必要首先在现有schema中查找并修改掉这种情况,除非确定这会导致问题。但是,如果计划在列上建索引,就应该尽量避免设计成可为NULL的列。
  当然也有例外,例如值得一提的是,InnoDB使用单独的位(bit)存储NULL值,所以对于稀疏数据有很好的空间效率。但这一点不适用于MyISAM。
  书中的描述说了几个主要问题,我这里暂且抛开MyISAM的问题不谈,这里我针对InnoDB作为考量条件。如果不设置NOTNULL的话,NULL是列的默认值,如果不是本身需要的话,尽量就不要使用NULL使用NULL带来更多的问题,比如索引、索引统计、值计算更加复杂,如果使用索引,就要避免列设置成NULL如果是索引列,会带来的存储空间的问题,需要额外的特殊处理,还会导致更多的存储空间占用对于稀疏数据有更好的空间效率,稀疏数据指的是很多值为NULL,只有少数行的列有非NULL值的情况默认值
  对于MySql而言,如果不主动设置为NOTNULL的话,那么插入数据的时候默认值就是NULL。
  NULL和NOTNULL使用的空值代表的含义是不一样,NULL可以认为这一列的值是未知的,空值则可以认为我们知道这个值,只不过他是空的而已。
  举个例子,一张表中的某一条name字段是NULL,我们可以认为不知道名字是什么,反之如果是空字符串则可以认为我们知道没有名字,他就是一个空值。
  而对于大多数程序的情况而言,没有什么特殊需要非要字段要NULL的吧,NULL值反而会对程序造成比如空指针的问题。
  对于现状大部分使用MyBatis的情况来说,我建议使用默认生成的insertSelective方法或者纯手动写插入方法,可以避免新增NOTNULL字段导致的默认值不生效或者插入报错的问题。值计算
  聚合函数不准确
  对于NULL值的列,使用聚合函数的时候会忽略NULL值。
  现在我们有一张表,name字段默认是NULL,此时对name进行count得出的结果是1,这个是错误的。
  count()是对表中的行数进行统计,count(name)则是对表中非NULL的列进行统计。
  失效
  对于NULL值的列,是不能使用表达式进行判断的,下面对name的查询是不成立的,必须使用isNULL。
  与其他值运算
  NULL和其他任何值进行运算都是NULL,包括表达式的值也是NULL。
  user表第二条记录age是NULL,所以1之后还是NULL,name是NULL,进行concat运算之后结果还是NULL。
  可以再看下下面的例子,任何和NULL进行运算的话得出的结果都会是NULL,想象下你设计的某个字段如果是NULL还不小心进行各种运算,最后得出的结果
  distinct、groupby、orderby
  对于distinct和groupby来说,所有的NULL值都会被视为相等,对于orderby来说升序NULL会排在最前
  其他问题
  表中只有一条有名字的记录,此时查询名字!a预期的结果应该是想查出来剩余的两条记录,会发现与预期结果不匹配。
  索引问题
  为了验证NULL字段对索引的影响,分别对name和age添加索引。
  关于网上很多说如果NULL那么不能使用索引的说法,这个描述其实并不准确,根据引用官方文档〔3〕里描述,使用isNULL和范围查询都是可以和正常一样使用索引的,实际验证的结果好像也是这样,看以下例子。
  然后接着我们往数据库中继续插入一些数据进行测试,当NULL列值变多之后发现索引失效了。
  我们知道,一个查询SQL执行大概是这样的流程:
  首先连接器负责连接到指定的数据库上,接着看看查询缓存中是否有这条语句,如果有就直接返回结果。
  如果缓存没有命中的话,就需要分析器来对SQL语句进行语法和词法分析,判断SQL语句是否合法。
  现在来到优化器,就会选择使用什么索引比较合理,SQL语句具体怎么执行的方案就确定下来了。
  最后执行器负责执行语句、有无权限进行查询,返回执行结果。
  从上面的简单测试结果其实可以看到,索引列存在NULL就会存在书中所说的导致优化器在做索引选择的时候更复杂,更加难以优化。存储空间
  数据库中的一行记录在最终磁盘文件中也是以行的方式来存储的,对于InnoDB来说,有4种行存储格式:REDUNDANT、COMPACT、DYNAMIC和COMPRESSED。
  InnoDB的默认行存储格式是COMPACT,存储格式如下所示,虚线部分代表可能不一定会存在。
  变长字段长度列表:有多个字段则以逆序存储,我们只有一个字段所有不考虑那么多,存储格式是16进制,如果没有变长字段就不需要这一部分了。
  NULL值列表:用来存储我们记录中值为NULL的情况,如果存在多个NULL值那么也是逆序存储,并且必须是8bit的整数倍,如果不够8bit,则高位补0。1代表是NULL,0代表不是NULL。如果都是NOTNULL那么这个就存在了。
  ROWID:一行记录的唯一标志,没有指定主键的时候自动生成的ROWID作为主键。
  TRXID:事务ID。
  ROLLPRT:回滚指针。
  最后就是每列的值。
  为了说明清楚这个存储格式的问题,我弄张表来测试,这张表只有c1字段是NOTNULL,其他都是可以为NULL的。
  可变字段长度列表:c1和c3字段值长度分别为1和2,所以长度转换为16进制是0x010x02,逆序之后就是0x020x01。
  NULL值列表:因为存在允许为NULL的列,所以c2,c3,c4分别为010,逆序之后还是一样,同时高位补0满8位,结果是00000010。
  其他字段我们暂时不管他,最后第一条记录的结果就是,当然这里我们就不考虑编码之后的结果了。
  这样就是一个完整的数据行数据的格式,反之,如果我们把所有字段都设置为NOTNULL,并且插入一条数据a,bb,ccc,dddd的话,存储格式应该这样:
  虽然我们发现NULL本身并不会占用存储空间,但是如果存在NULL的话就会多占用一个字节的标志位的空间。文章参考文档:
  https:dev。mysql。comdocrefman8。0enproblemswithnull。html
  https:dev。mysql。comdocrefman8。0enworkingwithnull。html
  https:dev。mysql。comdocrefman5。6enisnulloptimization。html
  https:dev。mysql。comdocrefman5。6eninnodbrowformat。html
  https:www。cnblogs。comzhoujinyiarticles2726462。html
  后台回复【pdf】获取百本计算机电子书和大厂面试精华,文章每周持续更新。我是艾小仙,阿里巴巴技术专家,我们下期见!
  面经PDF整理
投诉 评论 转载

为什么数据库字段要使用NOTNULL?最近刚入职新公司,发现数据库设计有点小问题,数据库字段很多没有NOTNULL,对于强迫症晚期患者来说,简直难以忍受,因此有了这篇文章。基于目前大部分的开发现状来说,我们都……棉被裹新车,保鲜膜包内饰,沈腾演绎的新人设好有味道2020年,东风本田联合开心麻花出品了微电影《沈浪的思薇》。男主沈浪追女主思薇追了14年,从滑板车到摩托再到轮椅,最后终于开上了新CRV,也终于追上了思薇。可是,当……3度争番位,撞衫蹭热度,她怎么还是红不起来?红颜未老恩先断,最是无情帝王家。加菲没有像大多宫妃一般爱上帝王,却为着白月光世子初见一笑远嫁异国,沦为棋子,受人摆布一生。加菲步步维艰爬上宫中权利顶峰,却惨遭白月光遗弃,……有一种媚感叫宋佳穿西装,透视内搭真挺撩人,简约款式也有魅力女性的穿着选择比较丰富,裤装裙装的搭配都能起到很强的作用,并且产生的魅力也大有不同。所以根据场合进行选择是很重要的,想要让个人的成熟感更强,还是清纯效果更强,都可以结合着服装进……盘点结婚生子后老得太快的八大女星,谁最让人心疼?对于娱乐圈的女明星来说结婚生孩子绝对是一项大的考验。谁都知道生孩子对于女人的身体影响是很大的,即便是再会保养的女明星也难逃结婚生子老得快的噩梦。下面我们就来盘点结婚生子后被吐槽……当红时抛下冯巩去当官,58岁家庭破裂,71岁的牛群如今怎样了离开了冯巩这么多年的牛群,到底作出了什么花样?2000年,彼时已经在全国爆火的牛群突然宣布要去一个叫蒙城县的地方当一个挂职的副县长,当时搭档冯巩听到这个消息大概想的是:这……唯美的诗歌只为你唯美的诗歌我就把你封存了一生一世只为你遗落在云中的那点胭红我就封存了春夏秋冬的妩媚春天那朵桃花缠绵在今生的无怨无悔春天那缕……家有皇帝妈妈,金榜题名女孩儿想带爸爸逃离亲妈的魔爪上帝不可能无处不在,所以他创造了妈妈,妈妈,提到这个词语都会让人感觉贴心、嘴角弯弯的温暖,她会最温柔地哄你开心、陪你长大。可是,丁楠楠她没有这样的妈妈。16岁的丁楠楠,花……十年痴恋英达却出轨,宋丹丹待巴图如生命,离婚为何选择ampa近年来,电视剧和电影在不断的发展,但发展最快的要数各类综艺。《王牌对王牌》作为综艺节目中的常青树,也是足足制作到了第五季。前段时间,作为曾经的小品女王宋丹丹也是带着……胡桃里音乐餐厅总经理,詹宗德呼吁抵制霸王餐活动胡桃里,去了会上瘾的地方。在那里,有着民谣的歌声。环境优美怡人,清新典雅。聚会,约会都很适合。好酒,好食,好景的衬托下,你会因此而陶醉,甚至忘记烦恼。2013年,胡……虎牙神豪遭主播辱骂!大哥豪刷过亿,仍遭主播嘲讽你算老几?专注吃瓜一百年!每日发布最新、最热、最有料的直播圈大事件!大型瓜田,欢迎各位观众老爷们持续关注提及虎牙神豪【汪汪汪】,相信时常关注直播的小伙伴一定不会感到陌生。作为……张若昀唐艺昕深夜公布恋情虐狗,阚清子纪凌尘大秀恩爱,迪丽热巴昨晚深夜炸弹当属张若昀与唐艺昕公布恋情,两人甜蜜亲吻,终于将这个人人都知道的秘密公开,实在虐狗!相恋六年实属不易啊!另一对阚清子纪凌尘也是在微博秀出恩爱的照片,俩人气质凸……
泰国贵妃上位记封妃废黜回宫被传当王后?漫威蜘蛛侠3格温将回归?艾玛斯通给出了个直接的回答现实更无底线!扫黑风暴徐英子原型被迫咬桌子,牙签扎私处95后女孩还原山海经神兽,被中国日报点赞爆美中国风宋朝灾民没钱买米,面对饥荒范仲淹说提高米价,事后才知其高明绿源核技战略再升级引发电动两轮车续航动力质变知否两个扶弟魔母亲损人不利己,最终,她们伤害了子女感情杜新枝想跟郭策重新做亲子鉴定,婆媳翻脸预警?从火遍全国到销声匿迹,大兵的错,不仅仅是因为调侃了黄家驹卢旺达饭店战争之后,数百万难民尸横遍野,发人深省岁月悠悠不会忘记,当年年轻人三件宝65式军帽军挎包军大衣章子怡汪峰为小苹果庆生,15岁少女肤白貌美,颜值赶超两个妈妈
原创就这一次我和我的倔强有什么好看的耽美小说(小众但惊艳的原耽推荐)道路安全会议记录模板2021沿着丝路去旅行哈萨克斯坦。科克托别山精选关于游戏的段落关于第一次感受失败的作文头衔通胀的状态是好还是坏?佛说善恶因果2012中国最具影响力的25位商界女性排行国宝大熊猫语文教案马斯克的星链在英国获得建地面站许可,提供全覆盖卫星互联网服务双飞造句用双飞造句大全

友情链接:中准网聚热点快百科快传网快生活快软网快好知文好找宜春榆林滕州锦州襄阳晋中泰州云南贵港许昌娄底河南乐清寿光阜阳汉中湖北漳州海口陕西延吉东营河源海南