iis服务器助手广告广告
返回顶部
首页 > 资讯 > 数据库 >为什么mysql字段要使用NOT NULL
  • 497
分享到

为什么mysql字段要使用NOT NULL

2024-04-02 19:04:59 497人浏览 薄情痞子
摘要

最近刚入职新公司,发现数据库设计有点小问题,数据库字段很多没有NOT NULL,对于强迫症晚期患者来说,简直难以忍受,因此有了这篇文章。 基于目前大部分的开发现状来说,我们都会把字段

最近刚入职新公司,发现数据库设计有点小问题,数据库字段很多没有NOT NULL,对于强迫症晚期患者来说,简直难以忍受,因此有了这篇文章。

基于目前大部分的开发现状来说,我们都会把字段全部设置成NOT NULL并且给默认值的形式。

  • 通常,对于默认值一般这样设置:
  • 整形,我们一般使用0作为默认值。
  • 字符串,默认空字符串

时间,可以默认1970-01-01 08:00:01,或者默认0000-00-00 00:00:00,但是连接参数要添加zeroDateTimeBehavior=convertToNull,建议的话还是不要用这种默认的时间格式比较好

但是,考虑下原因,为什么要设置成NOT NULL?

来自高性能Mysql中有这样一段话:

尽量避免NULL

很多表都包含可为NULL(空值)的列,即使应用程序并不需要保存NULL也是如此,这是因为可为NULL是列的默认属性。通常情况下最好指定列为NOT NULL,除非真的需要存储NULL值。

如果查询中包含可为NULL的列,对mysql来说更难优化,因为可为NULL的列使得索引、索引统计和值比较都更复杂。可为NULL的列会使用更多的存储空间,在Mysql里也需要特殊处理。当可为NULL的列被索引时,每个索引记录需要一个额外的字节,在MyISAM里甚至还可能导致固定大小的索引(例如只有一个整数列的索引)变成可变大小的索引。

通常把可为NULL的列改为NOT NULL带来的性能提升比较小,所以(调优时)没有必要首先在现有schema中查找并修改掉这种情况,除非确定这会导致问题。但是,如果计划在列上建索引,就应该尽量避免设计成可为NULL的列。

当然也有例外,例如值得一提的是,InnoDB使用单独的位(bit)存储NULL值,所以对于稀疏数据有很好的空间效率。但这一点不适用于MyISAM。

书中的描述说了几个主要问题,我这里暂且抛开MyISAM的问题不谈,这里我针对InnoDB作为考量条件。

  • 如果不设置NOT NULL的话,NULL是列的默认值,如果不是本身需要的话,尽量就不要使用NULL
  • 使用NULL带来更多的问题,比如索引、索引统计、值计算更加复杂,如果使用索引,就要避免列设置成NULL
  • 如果是索引列,会带来的存储空间的问题,需要额外的特殊处理,还会导致更多的存储空间占用
  • 对于稀疏数据又更好的空间效率,稀疏数据指的是很多值为NULL,只有少数行的列有非NULL值的情况

默认值

对于MySql而言,如果不主动设置为NOT NULL的话,那么插入数据的时候默认值就是NULL。

NULL和NOT NULL使用的空值代表的含义是不一样,NULL可以认为这一列的值是未知的,空值则可以认为我们知道这个值,只不过他是空的而已。

举个例子,一张表中的某一条name字段是NULL,我们可以认为不知道名字是什么,反之如果是空字符串则可以认为我们知道没有名字,他就是一个空值。

而对于大多数程序的情况而言,没有什么特殊需要非要字段要NULL的吧,NULL值反而会对程序造成比如空指针的问题。

对于现状大部分使用mybatis的情况来说,我建议使用默认生成的insertSelective方法或者纯手动写插入方法,可以避免新增NOT NULL字段导致的默认值不生效或者插入报错的问题。

值计算

聚合函数不准确

对于NULL值的列,使用聚合函数的时候会忽略NULL值。

现在我们有一张表,name字段默认是NULL,此时对name进行count得出的结果是1,这个是错误的。

count(*)是对表中的行数进行统计,count(name)则是对表中非NULL的列进行统计。

=失效

对于NULL值的列,是不能使用=表达式进行判断的,下面对name的查询是不成立的,必须使用is NULL

与其他值运算

NULL和其他任何值进行运算都是NULL,包括表达式的值也是NULL。

user表第二条记录age是NULL,所以+1之后还是NULL,name是NULL,进行concat运算之后结果还是NULL。

可以再看下下面的例子,任何和NULL进行运算的话得出的结果都会是NULL,想象下你设计的某个字段如果是NULL还不小心进行各种运算,最后得出的结果。。。

distinct、group by、order by

对于distinctgroup by来说,所有的NULL值都会被视为相等,对于order by来说升序NULL会排在最前

其他问题

表中只有一条有名字的记录,此时查询名字!=a预期的结果应该是想查出来剩余的两条记录,会发现与预期结果不匹配。

索引问题

为了验证NULL字段对索引的影响,分别对nameage添加索引。

关于网上很多说如果NULL那么不能使用索引的说法,这个描述其实并不准确,根据引用官方文档[3]里描述,使用is NULL和范围查询都是可以和正常一样使用索引的,实际验证的结果好像也是这样,看以下例子。

然后接着我们往数据库中继续插入一些数据进行测试,当NULL列值变多之后发现索引失效了。

我们知道,一个查询SQL执行大概是这样的流程:

首先连接器负责连接到指定的数据库上,接着看看查询缓存中是否有这条语句,如果有就直接返回结果。

如果缓存没有命中的话,就需要分析器来对SQL语句进行语法和词法分析,判断SQL语句是否合法。

现在来到优化器,就会选择使用什么索引比较合理,SQL语句具体怎么执行的方案就确定下来了。

最后执行器负责执行语句、有无权限进行查询,返回执行结果。

从上面的简单测试结果其实可以看到,索引列存在NULL就会存在书中所说的导致优化器在做索引选择的时候更复杂,更加难以优化。

存储空间

数据库中的一行记录在最终磁盘文件中也是以行的方式来存储的,对于InnoDB来说,有4种行存储格式:REDUNDANTCOMPACTDYNAMICCOMPRESSED

InnoDB的默认行存储格式是COMPACT,存储格式如下所示,虚线部分代表可能不一定会存在。

变长字段长度列表:有多个字段则以逆序存储,我们只有一个字段所有不考虑那么多,存储格式是16进制,如果没有变长字段就不需要这一部分了。

NULL值列表:用来存储我们记录中值为NULL的情况,如果存在多个NULL值那么也是逆序存储,并且必须是8bit的整数倍,如果不够8bit,则高位补0。1代表是NULL,0代表不是NULL。如果都是NOT NULL那么这个就存在了。

ROW_ID:一行记录的唯一标志,没有指定主键的时候自动生成的ROW_ID作为主键。

TRX_ID:事务ID。

ROLL_PRT:回滚指针。

最后就是每列的值。

为了说明清楚这个存储格式的问题,我弄张表来测试,这张表只有c1字段是NOT NULL,其他都是可以为NULL的。

可变字段长度列表:c1c3字段值长度分别为1和2,所以长度转换为16进制是0x01 0x02,逆序之后就是0x02 0x01

NULL值列表:因为存在允许为NULL的列,所以c2,c3,c4分别为010,逆序之后还是一样,同时高位补0满8位,结果是00000010

其他字段我们暂时不管他,最后第一条记录的结果就是,当然这里我们就不考虑编码之后的结果了。

这样就是一个完整的数据行数据的格式,反之,如果我们把所有字段都设置为NOT NULL,并且插入一条数据a,bb,ccc,DDDd的话,存储格式应该这样:

虽然我们发现NULL本身并不会占用存储空间,但是如果存在NULL的话就会多占用一个字节的标志位的空间。

文章参考文档:

https://dev.mysql.com/doc/refman/8.0/en/problems-with-null.html
Https://dev.mysql.com/doc/refman/8.0/en/working-with-null.html
https://dev.mysql.com/doc/refman/5.6/en/is-null-optimization.html
https://dev.mysql.com/doc/refman/5.6/en/innodb-row-fORMat.html
https://www.cnblogs.com/zhoujinyi/articles/2726462.html

到此这篇关于为什么mysql字段要使用NOT NULL的文章就介绍到这了,更多相关mysql字段使用NOT NULL内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

您可能感兴趣的文档:

--结束END--

本文标题: 为什么mysql字段要使用NOT NULL

本文链接: https://www.lsjlt.com/news/125805.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 为什么mysql字段要使用NOT NULL
    最近刚入职新公司,发现数据库设计有点小问题,数据库字段很多没有NOT NULL,对于强迫症晚期患者来说,简直难以忍受,因此有了这篇文章。 基于目前大部分的开发现状来说,我们都会把字段...
    99+
    2024-04-02
  • mysql如何将字段修改为not null
    本文小编为大家详细介绍“mysql如何将字段修改为not null”,内容详细,步骤清晰,细节处理妥当,希望这篇“mysql如何将字段修改为not null”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深...
    99+
    2024-04-02
  • MySQL中可为空的字段设置为NULL还是NOT NULL
    经常用mysql的人可能会遇到下面几种情况: 1、我字段类型是not null,为什么我可以插入空值 2、为什么not null的效率比null高 3、判断字段不为空的时候,到底要用 selec ...
    99+
    2024-04-02
  • MySQL字段为 NULL的5大坑
    目录1.count 数据丢失2.distinct 数据丢失3.select 数据丢失4.导致空指针异常5.增加了查询难度扩展知识:NULL 不会影响索引总结正式开始之前,我们先来看下 mysql 服务器的配置和...
    99+
    2023-04-27
    MySQL字段NULL MySQL NULL
  • MySQL字段为NULL的5大坑
    目录1.count 数据丢失2.distinct 数据丢失3.select 数据丢失4.导致空指针异常5.增加了查询难度扩展知识:NULL 不会影响索引总结正式开始之前,我...
    99+
    2023-05-18
    MySQL字段NULL MySQL NULL
  • 使用mybatis-plus想要修改某字段为null问题
    目录mybatis-plus想要修改某字段为null问题场景原因解决办法mybatis-plus更新字段为null不生效异常说明原理解决办法总结mybatis-plus想要修改某字段...
    99+
    2023-02-22
    mybatis-plus使用 修改某字段为null mybatis-plus修改字段
  • mysql怎么判断sql字段为null值
    mysql判断sql字段为null值的方法:mysql对null值处理提供了三个操作符:"is null"、"is not null"和<=>,而is null是列值为空返回true,is...
    99+
    2024-04-02
  • MySQL字段为NULL的坑有哪些
    这篇文章主要介绍“MySQL字段为NULL的坑有哪些”,在日常操作中,相信很多人在MySQL字段为NULL的坑有哪些问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”MySQL字...
    99+
    2023-05-12
    mysql null
  • mysql修改字段为null的方法
    这篇文章将为大家详细讲解有关mysql修改字段为null的方法,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。mysql修改字段为null的方法:1、新建一个查询,使用语句...
    99+
    2024-04-02
  • 什么是 MySQL NOT NULL 约束以及我们如何在创建表时声明字段 NOT NULL?
    实际上,MySQL NOT NULL 约束限制表的某一列具有 NULL 值。一旦我们对列应用了 NOT NULL 约束,那么我们就不能将空值传递给该列。它不能在整个表上声明,换句话说,我们可以说 NOT NULL 是列级约束。为了声明字段 ...
    99+
    2023-10-22
  • mysql字段为NULL索引会失效吗
    本篇内容介绍了“mysql字段为NULL索引会失效吗”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!项目场景:很多博客说mysql在字段中创建...
    99+
    2023-06-30
  • mysql的casewhen字段为空,null的问题
    目录mysql的case when字段为空,nullcase when 判断null(已测试)总结mysql的case when字段为空,null name字段为null时替换为 &...
    99+
    2022-12-14
    mysql的case when字段 case when字段为空 case when字段null
  • 使用mybatisplus修改某字段为null怎么解决
    本文小编为大家详细介绍“使用mybatisplus修改某字段为null怎么解决”,内容详细,步骤清晰,细节处理妥当,希望这篇“使用mybatisplus修改某字段为null怎么解决”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来...
    99+
    2023-07-05
  • Mybatis-Plus使用updateById()、update()将字段更新为null
    目录问题背景问题原因解决方案问题背景 昨晚同事找我帮他看一个问题,他使用mybatis-plus中提供的updateById方法,想将查询结果中某个字段原本不为null的值更新为nu...
    99+
    2022-11-13
    Mybatis-Plus 字段更新为null mybatis plus 更新为null
  • MySQL判断字段是否为null的方法
    这篇文章主要介绍MySQL判断字段是否为null的方法,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!我字段类型是not null,为什么我可以插入空值为毛not null的效率比nu...
    99+
    2024-04-02
  • mysql的case when字段为空,null的问题
    目录mysql的case when字段为空,nullcase when 判断null(已测试)总结mysql的case when字段为空,null name字段为null时替换为 ‘该字段为空’ ...
    99+
    2022-12-13
    mysql的casewhen字段 casewhen字段为空 casewhen字段null
  • sql怎么设置字段值为null
    要将SQL字段的值设置为NULL,可以使用以下方法:1. 在INSERT语句中使用NULL关键字来设置字段的值为NULL。例如:``...
    99+
    2023-10-12
    sql
  • MySQL中查询字段为空或者为null的方法
    目录mysql查询字段为空或者为null判断为null判断为空或空格MySQL查询字段为空(null)时设置默认值总结MySQL查询字段为空或者为null 判断为null select * from table...
    99+
    2022-12-12
    MySQL查询字段为空 MySQL查询字段为null MySQL查询字段
  • mysql中怎么避免使用null定义字段的原因是什么
    本篇文章给大家分享的是有关mysql中怎么避免使用null定义字段的原因是什么,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。一 NULL 为什么这么经常用(1) java的nu...
    99+
    2023-06-20
  • 为什么要使用MySQL
    这篇文章给大家分享的是有关为什么要使用MySQL的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。什么是MySQL  MySQL原本是一个开放源码的关系数据库管理系统,原开发者为瑞典...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作