mysql数据存储问题

目录

MySQL数据存储基础

MySQL数据存放位置

InnoDB存储引擎介绍

Mermaid图表:InnoDB存储引擎数据文件结构

表空间结构详解

组成要素

组织方式

页内组织

性能影响

Mermaid图表:表空间的层次化结构和页内组织

InnoDB行格式详解

行格式类型

Compact行格式详解

Mermaid图表:Compact行格式结构

变长字段和NULL值处理

变长字段长度列表

NULL值列表

Mermaid图表:变长字段和NULL值列表的存储逻辑

记录头信息详解

记录的组织方式:链表

记录状态

其他记录头信息

隐含字段

Mermaid图表:记录头信息结构

[特殊字段处理:VARCHAR(n) 字段大小限制](#特殊字段处理:VARCHAR(n) 字段大小限制)

大小限制处理

单字段情况下的VARCHAR(n)最大长度计算

多字段情况下的VARCHAR(n)最大长度计算

示例

[Mermaid图表:VARCHAR(n) 字段大小限制处理](#Mermaid图表:VARCHAR(n) 字段大小限制处理)

行溢出处理

行溢出的条件

行溢出的处理机制

不同行格式对行溢出的处理差异

Mermaid图表:行溢出处理机制

页与内存交互

页作为数据库与内存交互的基本单位

页的大小与作用

Mermaid图表:页与内存交互过程


MySQL数据存储基础

MySQL数据存放位置

MySQL的数据存储主要依赖于文件系统,数据文件的存放位置由配置文件中的datadir参数指定。每个数据库(Database)在datadir下创建一个以其名称命名的目录,例如/var/lib/mysql/mydatabase。在这个目录中,存储了该数据库的所有表结构和数据文件。

  • 数据目录( datadir ) : 这是MySQL数据文件存放的基础目录,通常在MySQL的配置文件my.cnfmy.ini中设置。

  • 表空间 文件 : 对于InnoDB存储引擎,表数据和索引通常存储在.ibd文件中,每个表可能有独立的表空间文件,文件名通常为table_name.ibd

InnoDB存储引擎介绍

InnoDB是MySQL的默认存储引擎,广泛用于需要事务支持和外键约束的场景。它提供了以下特性:

  • ACID事务: 保证了事务的原子性、一致性、隔离性和持久性。

  • 行级锁定: 提高了并发性能,只在修改行上加锁。

  • MVCC(多版本并发控制): 允许在不锁定资源的情况下读取数据,提高了读取性能。

InnoDB存储引擎的数据文件结构如下:

  • 系统 表空间 : 包含所有的表数据和索引,通常以ibdata1命名。

  • 独立 表空间 : 由innodb_file_per_table参数控制,每个表的数据和索引存储在独立的文件中,提高了空间管理的灵活性。

Mermaid图表:InnoDB存储引擎数据文件结构

这个Mermaid图表展示了InnoDB存储引擎的数据文件结构,其中系统表空间和独立表空间是两种主要的数据存储方式。

表空间结构详解

表空间(Tablespace)是InnoDB存储引擎中用于逻辑地组织和存储数据的概念。它通过一系列层次化的组件来优化数据的存储效率和访问速度。

组成要素
  1. 段(Segment):

    1. 表空间由多个段组成,每个段是数据的逻辑容器。

    2. 常见的段类型包括索引段(存储B+树索引的非叶子节点)和数据段(存储行数据)。

  2. 区(Extent):

    1. 区是段内由连续页组成的单元,每个区通常大小为1MB。

    2. 区的设计允许InnoDB引擎以连续的空间存储数据,减少磁盘寻址时间。

  3. 页(Page):

    1. 页是InnoDB存储引擎的基本存储单元,每个页默认大小为16KB。

    2. 页内可以存储多条行记录或索引数据,是实际数据存储的地方。

  4. 行(Row):

    1. 行是页内存储的数据记录,每条记录根据其数据类型和长度有不同的存储方式。
组织方式

表空间的组织方式具有以下特点:

  • 层次化存储: 数据通过段、区、页的层次化结构进行存储,每一层都服务于不同的目的和优化。

  • 连续性优化: 通过区的使用,InnoDB能够将数据存储在物理上连续的页中,这有助于提高I/O效率。

  • 灵活的行存储: 行记录根据其大小和类型存储在页内,如果一行数据过大,可能会发生行溢出,即数据被存储在多个页中。

页内组织
  • 每个页可以存储多条行记录,行记录按照它们在页内的插入顺序进行组织。

  • 页内还包含一些控制信息,如页头、页目录和页尾,这些信息帮助管理页内的数据。

性能影响
  • 表空间的结构对数据库的性能有直接影响。例如,区的使用可以减少磁盘随机I/O操作,而页的大小直接影响每次I/O操作可以读取的数据量。
Mermaid图表:表空间的层次化结构和页内组织

mysql数据存储问题这个Mermaid图表详细展示了表空间的层次化结构,以及页内如何组织行记录和控制信息。通过这种结构,InnoDB存储引擎能够高效地管理大量数据,并提供快速的数据访问能力。

InnoDB行格式详解

InnoDB存储引擎提供了多种行格式,以适应不同的存储需求和优化性能。每种行格式都有其特定的结构和特点。

行格式类型
  1. Redundant: 这是MySQL早期版本使用的行格式,现在已不推荐使用,因为它会占用更多的存储空间。

  2. Compact: 这种格式旨在节省空间,是MySQL 5.1版本之后默认的行格式。

  3. Dynamic: 动态行格式适用于有大量变长字段的表,提供了更灵活的存储方式。

  4. Compressed: 压缩行格式使用 zlib 压缩库来减少数据的存储大小。

Compact行格式详解

Compact行格式是当前默认的行格式,它通过以下方式组织数据:

  1. 记录的额外信息:

    1. 变长字段长度列表 : 对于变长字段(如VARCHARVARBINARY等),存储实际占用的字节数。

    2. NULL值列表: 如果表定义中字段可以为NULL,使用位图来标记哪些字段的值为NULL。

  2. 记录头信息:

    1. 包含行的状态信息,如是否被删除、记录的类型等。
  3. 记录的真实数据:

    1. 存储字段数据,包括显式字段和隐含字段。

    2. 显式字段是用户定义的列数据。

    3. 隐含字段是InnoDB自动添加的,包括:

      • row_id: 如果表没有定义主键或唯一非空索引,InnoDB使用隐藏的自增字段row_id

      • trx_id: 事务ID,标识生成该行记录的事务。

      • roll_pointer: 回滚指针,用于事务的多版本并发控制(MVCC)。

Mermaid图表:Compact行格式结构

这个Mermaid图表展示了Compact行格式的结构,包括记录的额外信息和真实数据的组成部分。通过这种结构,InnoDB能够有效地存储和管理行数据,同时提供对事务和并发控制的支持。

变长字段和NULL值处理

在InnoDB的Compact和Dynamic行格式中,变长字段和NULL值的处理对存储效率和数据访问速度有重要影响。

变长字段长度列表

作用:

  • 变长字段(如VARCHARVARBINARY等)在存储时,其长度是可变的。变长字段长度列表用于记录每个变长字段实际占用的字节数。

存储方式:

  • 对于每个变长字段,InnoDB会根据其实际长度分配相应的字节来存储长度信息。

  • 如果变长字段的最大长度小于或等于255字节,使用1字节来存储长度。

  • 如果变长字段的最大长度大于255字节,使用2字节来存储长度。

优化:

  • 变长字段长度列表通常存储在记录的开始部分,这样在读取记录时可以快速确定每个变长字段的大小。
NULL值列表

存储逻辑:

  • 如果表中的字段定义允许NULL值,InnoDB使用NULL值列表来记录哪些字段的值为NULL。

位图 表示:

  • NULL值列表是一个位图,每个字段对应位图中的一个位。如果位的值为1,则表示该字段的值为NULL;如果为0,则表示有具体值。

存储效率:

  • 使用位图表示NULL值可以节省空间,特别是当表中有大量可以为NULL的字段时。

特殊情况:

  • 如果表中所有字段都定义为NOT NULL,NULL值列表将不会被包含在行格式中,从而节省存储空间。
Mermaid图表:变长字段和NULL值列表的存储逻辑

图表展示了变长字段长度列表的存储方式和NULL值列表的存储逻辑。通过这种方式,InnoDB行格式能够有效地处理变长字段和NULL值,同时保持存储的紧凑性和访问的高效性。

记录头信息详解

记录头信息是InnoDB行格式中的重要组成部分,它包含了记录的元数据和状态信息,对于数据库的事务处理和数据访问至关重要。

记录的组织方式:链表

InnoDB使用链表来组织记录,每个记录头信息中包含指向下一条记录的指针。这种链表结构使得InnoDB能够高效地进行以下操作:

  • 插入: 新记录可以快速地插入到链表的适当位置。

  • 删除 : 删除操作可以通过修改记录头信息中的delete_mask来标记记录为删除状态,而不需要立即从磁盘上移除记录。

  • 更新: 当记录更新时,如果新数据导致记录大小变化,可能需要将记录移动到新的位置。

记录状态

记录头信息中的状态字段包括但不限于:

  • delete_mask : 用于标记记录是否被删除。如果一个记录被删除,delete_mask将被设置,但实际上记录的数据仍然保留在页中,直到被垃圾收集器清理。

  • record_type: 指示记录的类型。不同的值代表不同类型的记录,例如:

    • 0 表示普通记录。

    • 1 表示B+树非叶子节点记录。

    • 2 表示最小记录,用于索引。

    • 3 表示最大记录,用于索引。

其他记录头信息

除了上述状态信息,记录头还包含:

  • next_record: 下一条记录的地址,用于实现链表的物理连接。

  • heap_number: 记录在堆中的序号,用于在二级索引中快速定位记录。

隐含字段

记录头信息中还包括对隐含字段的引用,如:

  • row_id : 如果表没有主键或唯一索引,InnoDB会使用内部的自增row_id作为隐含的主键。

  • trx_id: 记录创建或最后修改该记录的事务ID。

  • roll_pointer: 回滚指针,用于MVCC和事务的回滚操作。

Mermaid图表:记录头信息结构

这个Mermaid图表展示了记录头信息的结构,包括记录的组织方式、状态信息和隐含字段。这些信息对于InnoDB存储引擎的高效数据管理和事务处理至关重要。

特殊字段处理:VARCHAR(n) 字段大小限制

VARCHAR(n) 字段在InnoDB中具有特殊的处理方式,因为它是一个变长字段,其实际存储长度取决于存储的数据。

大小限制处理
  1. 最大长度限制:

    1. VARCHAR(n) 字段的最大长度n受到表的行大小限制影响。一行数据的大小(包括所有的字段和额外的存储开销)不能超过65535字节。
  2. 存储开销:

    1. 行的大小限制包括了变长字段长度列表和NULL值列表的存储开销。
  3. 变长字段长度列表:

    1. 对于VARCHAR字段,InnoDB需要额外的字节来存储字段的实际长度。如果字段长度小于或等于255字节,使用1字节存储;如果超过255字节,则使用2字节存储。
  4. NULL值列表:

    1. 如果表中有字段可以为NULL,InnoDB会使用一个位图(NULL值列表)来标记哪些字段是NULL,这占用了额外的空间。
单字段情况下的VARCHAR(n)最大长度计算
  • 当表中只有一个VARCHAR字段时,最大长度n可以通过以下公式计算得出: nmax=65535−(存储开销)nmax=65535−(存储开销) 其中存储开销包括变长字段长度列表和NULL值列表的字节数。
多字段情况下的VARCHAR(n)最大长度计算
  • 当表中有多个字段,包括VARCHAR和其他类型字段时,需要考虑所有字段的总大小和存储开销。VARCHAR字段的最大长度n需要保证整个行的大小不超过65535字节。
示例
  • 假设一个表只有一个VARCHAR字段,并且该字段允许为NULL。在这种情况下,行的存储开销包括1字节的NULL值列表和2字节的变长字段长度列表(假设VARCHAR字段长度超过255字节): nmax=65535−2−1=65532nmax=65535−2−1=65532 这意味着在这种情况下,VARCHAR字段的最大长度可以是65532字节。
Mermaid图表:VARCHAR(n) 字段大小限制处理

这个Mermaid图表展示了VARCHAR(n)字段大小限制的处理方式,包括单字段和多字段情况下的最大长度计算方法。这种处理确保了VARCHAR字段在InnoDB中的高效存储和灵活使用。

行溢出处理

在InnoDB存储引擎中,当一行数据的大小超过了页的大小限制时,就会发生行溢出。这种情况需要特殊的处理机制来确保数据的完整性和访问效率。

行溢出的条件
  • 当一行记录的大小超过页的允许大小(通常是16KB)时,该行就会发生溢出。

  • 溢出通常发生在VARCHARTEXTBLOB等大字段上,尤其是当这些字段的值较大时。

行溢出的处理机制
  1. 溢出页(Overflow Page):

    1. 超出一页大小的数据会被存储在额外的溢出页中。

    2. 这些溢出页通过指针与原始记录页相连。

  2. 记录头信息:

    1. 在记录头信息中,会有指向溢出页的指针,以便能够找到溢出的数据。
  3. 数据存储:

    1. 对于Compact和Dynamic行格式,行溢出时,原始页中会保留一部分数据,而其余数据存储在溢出页中。

    2. 对于Compressed行格式,行溢出时,原始页中不保留数据,所有溢出数据都存储在溢出页中。

不同行格式对行溢出的处理差异
  • Compact和Dynamic行格式:

    • 行溢出时,原始记录页保留数据的前768字节,其余数据存储在溢出页中。

    • 这种处理方式允许在不访问溢出页的情况下快速访问行的前768字节。

  • Compressed行格式:

    • 行溢出时,原始记录页不保留任何数据,而是使用20字节的指针指向溢出页。

    • 这种处理方式减少了原始记录页的使用,但访问溢出数据时需要额外的I/O操作。

Mermaid图表:行溢出处理机制

这个Mermaid图表展示了行溢出的条件和处理机制,以及不同行格式对行溢出处理的差异。通过这种机制,InnoDB能够高效地处理大型数据字段,同时保持数据访问的性能。

页与内存交互

在数据库系统中,页是数据存储和内存交互的基本单位,对于InnoDB存储引擎来说,页的使用至关重要。

页作为数据库与内存交互的基本单位
  1. 数据分页存储:

    1. 数据库将数据分页存储,每页固定大小,通常是16KB,这个大小远大于操作系统的内存页大小,这样做可以减少I/O操作的次数。
  2. 内存 页缓存:

    1. 数据库管理系统(DBMS)会在内存中缓存一部分页,以加速数据访问。当查询涉及到某页的数据时,如果该页已经在内存中,则可以直接从内存中读取,否则需要从磁盘中加载。
  3. 页的加载与卸载:

    1. 当内存中的页缓存满了之后,DBMS会根据特定的算法(如LRU算法)来决定哪些页应该被卸载到磁盘,哪些页应该保留在内存中。
页的大小与作用
  1. 页的大小:

    1. InnoDB的页大小默认是16KB,这个大小是经过优化的,以适应现代硬件的块设备和文件系统。
  2. 页的作用:

    1. 页作为数据访问的基本单位,可以存储多条行记录或B+树的多个节点。

    2. 页的大小直接影响到I/O效率,较大的页可以减少访问单条记录所需的I/O次数,但可能会造成空间的浪费。

  3. 页的组成:

    1. 每个页包含页头、页目录、行记录、页尾等部分。页头包含页的元数据,页目录用于快速定位到页内的特定行,页尾包含页的状态信息。
  4. 页的I/O操作:

    1. 数据库的读取(Read)和写入(Write)操作通常以页为单位进行,这样可以减少磁盘I/O的次数,提高性能。
Mermaid图表:页与内存交互过程

这个Mermaid图表展示了页作为数据库与内存交互的基本单位的过程,以及页的大小和作用。通过这种方式,InnoDB能够有效地管理数据的存储和访问,优化性能。

总结

1. MySQL NULL值存放方式
  • NULL值列表: 在InnoDB的Compact和Dynamic行格式中,如果表结构允许字段包含NULL值,InnoDB会使用一个NULL值列表(位图)来记录哪些字段是NULL。这个列表是按列的逆序排列的,每个位对应一个字段,1表示NULL,0表示非NULL。

  • 空间节省: 当表中的所有字段都设置为NOT NULL时,NULL值列表不会被包含在行格式中,从而节省存储空间。

2. VARCHAR字段大小计算方法
  • 最大长度限制 : VARCHAR(n)字段的大小受限于行的大小限制,即一行数据的最大字节数为65535字节。

  • 存储开销: 这个最大长度包括了变长字段长度列表和NULL值列表的存储开销。变长字段长度列表的存储取决于字段实际使用的最大长度,而NULL值列表至少占用1字节。

  • 计算公式 : 对于允许为NULL的VARCHAR字段,其最大长度n的计算公式是: nmax=65535−(变长字段长度列表字节数+NULL值列表字节数)nmax=65535−(变长字段长度列表字节数+NULL值列表字节数)

  • 字符集影响 : 字符集的不同会影响每个字符所占用的字节数,进而影响VARCHAR字段的最大长度。

3. 行溢出后MySQL的处理策略
  • 溢出页: 当一行数据的大小超过页的大小限制时,超出部分会被存储在额外的溢出页中。

  • 不同行格式的处理:

    • Compact和Dynamic: 行的前768字节(如果足够容纳)保留在原始页中,其余部分存储在溢出页。

    • Compressed: 行的所有数据(除了指针)存储在溢出页中,原始页仅包含指向溢出数据的20字节指针。

  • 性能考虑: 行溢出会增加访问数据所需的I/O操作次数,因为需要从不同的页读取数据。

Mermaid图表:MySQL行存储策略总结

这个Mermaid图表总结了MySQL中NULL值的存放方式、VARCHAR字段大小的计算方法,以及行溢出后的处理策略。这些策略共同确保了数据的有效存储和高效访问。

相关推荐
明月看潮生2 分钟前
青少年编程与数学 02-007 PostgreSQL数据库应用 11课题、视图的操作
数据库·青少年编程·postgresql·编程与数学
阿猿收手吧!9 分钟前
【Redis】Redis入门以及什么是分布式系统{Redis引入+分布式系统介绍}
数据库·redis·缓存
奈葵13 分钟前
Spring Boot/MVC
java·数据库·spring boot
leegong2311121 分钟前
Oracle、PostgreSQL该学哪一个?
数据库·postgresql·oracle
中东大鹅27 分钟前
MongoDB基本操作
数据库·分布式·mongodb·hbase
夜光小兔纸1 小时前
Oracle 普通用户连接hang住处理方法
运维·数据库·oracle
涅槃寂雨2 小时前
C语言小任务——寻找水仙花数
c语言·数据结构·算法
『往事』&白驹过隙;2 小时前
操作系统(Linux Kernel 0.11&Linux Kernel 0.12)解读整理——内核初始化(main & init)之缓冲区的管理
linux·c语言·数据结构·物联网·操作系统
就爱学编程2 小时前
从C语言看数据结构和算法:复杂度决定性能
c语言·数据结构·算法
半桔2 小时前
栈和队列(C语言)
c语言·开发语言·数据结构·c++·git