服务器之家:专注于服务器技术及软件下载分享
分类导航

Mysql|Sql Server|Oracle|Redis|MongoDB|PostgreSQL|Sqlite|DB2|mariadb|Access|数据库技术|

服务器之家 - 数据库 - Mysql - MySQL 的 NULL 值是怎么存放的?

MySQL 的 NULL 值是怎么存放的?

2023-10-27 01:01未知服务器之家 Mysql

InnoDB页 InnoDB是一个将数据存储到磁盘上的存储引擎,所以就算我们关闭、重启服务器,数据还是存在的。而在真正处理数据的时候是在内存中进行的,所以需要把磁盘中的内容加载到内存中。 我们知道读写磁盘是很慢的。当我们

InnoDB页

MySQL 的 NULL 值是怎么存放的?

InnoDB是一个将数据存储到磁盘上的存储引擎,所以就算我们关闭、重启服务器,数据还是存在的。而在真正处理数据的时候是在内存中进行的,所以需要把磁盘中的内容加载到内存中。

我们知道读写磁盘是很慢的。当我们想从表里获取数据的时候,InnoDB会一条一条的从磁盘中读出来吗?不会的!因为那样太慢了。它采取的方式是:将数据划分为若干页,以页做为磁盘和内存交互的基本单位。InnoDB中页的大小一般为16KB。

在服务器运行的过程中不可以修改页的大小,只能在初始化数据目录的时候指定。

InnoDB 行格式

行格式有哪些

行格式(row_format):一条数据记录在磁盘上的存储结构。

InnoDB 提供了 4 种行格式,分别是 Redundant、Compact、Dynamic和 Compressed 行格式。

我们可以在创建表或者修改表的语句中指定所使用的行格式

create table 'table info ..' row_format = '行格式名称'
alter table 'table name' row_format = '行格式名称'
  • Redundant:是很古老的行格式了, MySQL 5.0 版本之前用的行格式,现在基本没人用了。
  • Compact:由于 Redundant 不是一种紧凑的行格式,所以 MySQL 5.0 之后引入了 Compact 行记录存储方式,Compact 是一种紧凑的行格式,设计的初衷就是为了让一个数据页中可以存放更多的行记录,从 MySQL 5.1 版本之后,行格式默认设置成 Compact。
  • Dynamic 和 Compressed 两个都是紧凑的行格式,它们的行格式都和 Compact 差不多,因为都是基于 Compact 改进一点东西。从 MySQL5.7 版本之后,默认使用 Dynamic 行格式。

Redundant 行格式因为现在基本没人用了,重点介绍 Compact 行格式,因为 Dynamic 和 Compressed 这两个行格式跟 Compact 非常像。

Compact 格式

话不多说,直接看图

MySQL 的 NULL 值是怎么存放的?

记录额外的信息

这部分信息是服务器为了更好的管理记录而不得不额外添加的一些信息,这些额外信息分为三个部分,分别是:变长字段长度列表、NULL值列表和记录头信息。

变长字段长度列表

在mysql中有一些变长的数据类型,比如varchar( )、varbinary( )、text类型、blob类型,我们把使用这个变长类型的列成为变长字段。

所以,在存储数据的时候,也要把数据占用的大小存起来,存到「变长字段长度列表」里面,读取数据的时候才能根据这个「变长字段长度列表」去读取对应长度的数据。

这些变长字段的真实数据占用的字节数会按照列的顺序逆序存放(后面会说为什么要这么设计)。

为了展示具体是怎么保存「变长字段的真实数据占用的字节数」,我们先创建这样一张表,字符集是 ascii(所以每一个字符占用的 1 字节),行格式是 Compact,student 表中 name 和 dream_school 字段是变长字段:

CREATE TABLE `student` (
  `id` int(11) NOT NULL,
  `name` VARCHAR(20) DEFAULT NULL,
  `dream_school` VARCHAR(20) DEFAULT NULL,
  `age` int(11) DEFAULT NULL,
  PRIMARY KEY (`id`) USING BTREE
) ENGINE = InnoDB DEFAULT CHARACTER SET = ascii ROW_FORMAT = COMPACT;

我们插入三条记录

MySQL 的 NULL 值是怎么存放的?

我们看看这三条记录的行格式中的 变长字段长度列表 是怎样存储的。

先来看第一条记录:

  • name 列的值为 1,真实数据占用的字节数是 1 字节,十六进制 0x01;
  • dream_school 列的值为 qinghua,真实数据占用的字节数是 7 字节,十六进制 0x07;
  • age 列和 id 列不是变长字段,这里不用管。

MySQL 的 NULL 值是怎么存放的?

再来看第二条

  • name 列的值为 1,真实数据占用的字节数是 2 字节,十六进制 0x02;
  • dream_school 列的值为 beida,真实数据占用的字节数是 5 字节,十六进制 0x05;

MySQL 的 NULL 值是怎么存放的?

第三条记录

第三条记录 dream_school 列的值是 NULL,NULL 是不会存放在行格式中记录的真实数据部分里的。

MySQL 的 NULL 值是怎么存放的?

null值列表一条记录中的某些列可能存储 NULL 值,如果把这些 NULL 值都放到记录的 真实数据中存储会很占地方,所以 OMPACT 行格式把一条记录中值为 NULL 的列统一管理起来,存储到 NULL 值列表中.

处理过程:

  • 首先统计表中允许存储 NULL 的列有哪些,主键列以及使用 NOT NULL 修饰的列都是不可以存储 NULL 值的,所以在统计的时候不会把这些列算进去。
  • 如果表中没有允许存储 NULL 的列,则 NULL 值列表也就不存在了,否则将每个允许存储 NULL 的列对应一个 进制位,二进制位按照列的顺序逆序排列。二进制位表示的意义如下
  • 进制位的值为1时,代表该列的值为NULL。
  • 迸制位的值为0时,代表该列的值不为NULL。
  • 另外,NULL 值列表必须用整数个字节的位表示(1字节8位),如果使用的二进制位个数不足整数个字节,则在字节的高位补 0。

先来看第一条记录

第一条记录所有列都有值,不存在 NULL 值,用二进制来表示是这样的:

MySQL 的 NULL 值是怎么存放的?

第二条记录

接下来看第二条记录,第二条记录 age 列是 NULL 值,用二进制来表示是这样的:

MySQL 的 NULL 值是怎么存放的?

第三条记录

第三条记录 dream_school 列 和 age 列是 NULL 值,用二进制来表示是这样的:

MySQL 的 NULL 值是怎么存放的?

记录头信息

记录头信息由固定5个字节组成,用于描述记录的一些属性,这里的属性比较多,我们就列举几个相对来说重要点的。

  • deleted_flag :标识此条数据是否被删除。从这里可以知道,我们执行 detele 删除记录的时候,并不会真正的删除记录,只是将这个记录的 delete_mask 标记为 1。
  • next_record:下一条记录的位置。所以我们可以知道,记录与记录之间是通过链表组织的。在前面我也提到了,指向的是下一条记录的「记录头信息」和「真实数据」之间的位置,这样的好处是向左读就是记录头信息,向右读就是真实数据,比较方便。
  • record_type:表示当前记录的类型,0表示普通记录,1表示B+树非叶子节点记录,2表示最小记录,3表示最大记录

MySQL 的 NULL 值是怎么存放的?

记录真实数据

记录真实数据除了记录我们自定义的列的数据外,Mysql还会为每个记录默认添加一些列(隐藏列)

MySQL 的 NULL 值是怎么存放的?

  • row_id:当我们创建表的时候没有指定主键,也没有唯一约束的列,innodb 就会自动的为这些记录添加row_id隐藏字段,占用6个字节。不是必须会有的。
  • trx_id:事务ID,这个列是必须的,占用6个字节。表示数据是有哪个事务生产的。
  • roll_pointer:回滚指针,表示当前记录上一个版本的指针,这个列也是必需的,占用 7 个字节。

MVCC机制就是依赖 trx_id 和 roll_pointer 来实现的。

行溢出后,MySQL 是怎么处理的?

MySQL 中磁盘和内存交互的基本单位是页,一个页的大小一般是 16KB,也就是 16384字节,而一个 varchar(n) 类型的列最多可以存储 65532字节,一些大对象如 TEXT、BLOB 可能存储更多的数据,这时一个页可能就存不了一条记录。这个时候就会发生行溢出,多的数据就会存到另外的「溢出页」中。

如果一个数据页存不了一条记录,InnoDB 存储引擎会自动将溢出的数据存放到「溢出页」中。在一般情况下,InnoDB 的数据都是存放在 「数据页」中。但是当发生行溢出时,溢出的数据会存放到「溢出页」中。

当发生行溢出时,在记录的真实数据处只会保存该列的一部分数据,而把剩余的数据放在「溢出页」中,然后真实数据处用 20 字节存储指向溢出页的地址,从而可以找到剩余数据所在的页。大致如下图所示。

MySQL 的 NULL 值是怎么存放的?

总结

MySQL 的 NULL 值是怎么存放的?

MySQL 的 Compact 行格式中会用「NULL值列表」来标记值为 NULL 的列,NULL 值并不会存储在行格式中的真实数据部分。

NULL值列表会占用 1 字节空间,当表中所有字段都定义成 NOT NULL,行格式中就不会有 NULL值列表,这样可节省 1 字节的空间。

行溢出后,MySQL 是怎么处理的?

如果一个数据页存不了一条记录,InnoDB 存储引擎会自动将溢出的数据存放到「溢出页」中。

Compact 行格式针对行溢出的处理是这样的:当发生行溢出时,在记录的真实数据处只会保存该列的一部分数据,而把剩余的数据放在「溢出页」中,然后真实数据处用 20 字节存储指向溢出页的地址,从而可以找到剩余数据所在的页。


延伸 · 阅读

精彩推荐
  • MysqlMySQL 数据备份与还原的示例代码

    MySQL 数据备份与还原的示例代码

    这篇文章主要介绍了MySQL 数据备份与还原的相关知识,本文通过示例代码给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下...

    逆心2972019-06-23
  • MysqlERROR: Error in Log_event::read_log_event()

    ERROR: Error in Log_event::read_log_event()

    ERROR: Error in Log_event::read_log_event(): read error, data_len: 438, event_type: 2 ...

    MYSQL教程网6412020-03-13
  • Mysql解决MySQl查询不区分大小写的方法讲解

    解决MySQl查询不区分大小写的方法讲解

    今天小编就为大家分享一篇关于解决MySQl查询不区分大小写的方法讲解,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起...

    Veir_dev5592019-06-25
  • MysqlMySQL数据库varchar的限制规则说明

    MySQL数据库varchar的限制规则说明

    本文我们主要介绍了MySQL数据库中varchar的限制规则,并以一个实际的例子对限制规则进行了说明,希望能够对您有所帮助。 ...

    mysql技术网4192019-11-23
  • Mysql详解MySQL中的分组查询与连接查询语句

    详解MySQL中的分组查询与连接查询语句

    这篇文章主要介绍了MySQL中的分组查询与连接查询语句,同时还介绍了一些统计函数的用法,需要的朋友可以参考下 ...

    GALAXY_ZMY5442020-06-03
  • MysqlMySQL锁的知识点总结

    MySQL锁的知识点总结

    在本篇文章里小编给大家整理了关于MySQL锁的知识点总结以及实例内容,需要的朋友们学习下。...

    别人放弃我坚持吖4362020-12-14
  • Mysql浅谈mysql 树形结构表设计与优化

    浅谈mysql 树形结构表设计与优化

    在诸多的管理类,办公类等系统中,树形结构展示随处可见,本文主要介绍了mysql 树形结构表设计与优化,具有一定的参考价值,感兴趣的小伙伴们可以参...

    小码农叔叔5242021-11-16
  • Mysqlmysql 不能插入中文问题

    mysql 不能插入中文问题

    当向mysql5.5插入中文时,会出现类似错误 ERROR 1366 (HY000): Incorrect string value: '\xD6\xD0\xCE\xC4' for column ...

    MYSQL教程网5722019-11-25