MySQL数据结构选择

embedded/2025/1/8 22:58:26/

系列文章目录

一、MySQL数据结构选择
二、MySQL性能优化explain关键字详解
三、MySQL索引优化


文章目录

  • 系列文章目录
  • 前言
  • 一、索引
    • 1.1、什么是索引
    • 1.2、构建索引的过程
    • 1.3、索引的更新和维护
    • 1.4、索引的查询和管理
    • 1.5、InnoDB 和 MyISAM 的索引实现
    • 1.6、联合索引和最左前缀法则
  • 二、MySQL数据结构选择
    • 2. 1、二叉树(平衡二叉树)
    • 2.2、红黑树
    • 2.3、Hash表
    • 2.4、B树
    • 2.5、B+树


前言

  本篇着重剖析MySQL索引,以及底层关于索引数据结构的选择。
  数据结构网址:数据结构


一、索引

1.1、什么是索引

  索引类似于一本书的目录,可以让数据库在检索数据时快速定位目标行,而不需要逐一扫描整个表。假设需要在《计算机网络》这本书中找到数据链路层(某一行的数据)这一章,如果不通过目录(索引),则需要一页一页地去翻,直到找到目标为止。

1.2、构建索引的过程

  构建索引的过程,实际上是将数据库中的一列或多列的数据按照特定规则构建成一种有序数据结构,构建索引的过程,可以分为以下的步骤:

  1. 收集列的数据: MySQL 读取需要构建索引的列的数据。(如果是主键索引,MySQL 会读取整个主键列的数据)
  2. 排序数据: 将列的数据按照值的大小排序,排序是为了在后续查找过程中减少搜索范围。
  3. 构建树形结构: 以下文会提及的B+树数据结构为例,首先会将排序后的数据分成多个小块,每个块称为一个页(Page),然后会构建树形结构,最底层的节点称为叶子节点,每个叶子节点存储一个范围的数据。叶子节点之间按顺序通过双向链表连接,便于范围查询,上一层节点会存储下一层节点中的关键数据(最小值)。
  4. 将索引结构存储到磁盘: 索引的每个节点存储在磁盘的特定页中,以供后续查询。
    B+树  也就是说,索引是帮助Mysql高效获取数据的排好序的数据结构

1.3、索引的更新和维护

  在数据表发生增删改操作时,索引也需要维护:

  • 插入数据:将新值插入到索引结构中,可能需要分裂节点。(每个节点有大小限制,超过大小则会触发分裂)
  • 删除数据:从索引中移除对应值,可能会合并节点以保持平衡。
  • 更新数据:更新索引列时,等价于删除旧索引和插入新索引。

  所以在单体架构下,推荐使用整型的自增主键,顺序插入以避免频繁地发生节点分裂。至于为什么推荐使用int类型,而不是UUID,是因为在构建B+树的结构时,确定新的节点应该在目标节点的左侧/右侧。数值之间的比较,速度远大于字符串之间的比较。(字符串的比较,是将完整的字符串拆成单个字符,然后转换成ASCII码进行比较)

  索引在加速查询的同时,一定程度上也会影响插入和删除的效率,所以索引并非越多越好

1.4、索引的查询和管理

  1. 查看表中的索引:
SHOW INDEX FROM table_name;
  1. 创建索引:
-- 普通索引
CREATE INDEX idx_column_name ON table_name(column_name);-- 联合索引
CREATE INDEX idx_name_age ON users(name, age);-- 唯一索引
CREATE UNIQUE INDEX idx_unique_name ON users(name);
  1. 删除索引:
DROP INDEX idx_column_name ON table_name;

1.5、InnoDB 和 MyISAM 的索引实现

  MyISAM 的索引文件和数据文件是分离的,也就是MyISAM是非聚集索引。其索引文件存放在*.MYI中,数据文件存放在*.MYD中。而InnoDB 的索引实现是聚集的,也就是表数据文件本身就是按B+Tree组织的一个索引结构文件。

1.6、联合索引和最左前缀法则

  假设我某张表建立了一个name,age,address的联合索引,同时表中有如下数据,构建索引的过程如下:

| name | age | address |
|--------|-----|-----------------|
| Alice | 30 | New York |
| Bob | 25 | California |
| Alice | 25 | Los Angeles |
| Bob | 30 | Chicago |

  则联合索引键按 name, age, address 的顺序将数据组织成以下结构:

(Alice, 25, Los Angeles)
(Alice, 30, New York)
(Bob, 25, California)
(Bob, 30, Chicago)

  首先对联合索引中的第一个字段的值进行排序,相同的则对二,三字段依次排序,根据这样的排序规则,很容易理解,为何联合索引的最左前缀法则中,必须要左到右的连续列进行查询,如果跳过了一列,则无法利用后续列。例如,对于联合索引 (name, age, address),可以使用:name name, agename, age, address,不能单独使用:age address:根本原因在于,在构建索引时,排序是按照联合索引从左到右的顺序依次进行的,如果跳过了最左列,那后续的字段必然是没有排序完成的,即只有在明确 name 的值后,才能进一步判断ageaddress 的值。
在这里插入图片描述

二、MySQL数据结构选择

  树的高度与节点访问次数直接相关,每次查询都需要从根节点开始,一层一层地向下访问,直到找到目标叶子节点。而磁盘的访问特点是,随机访问速度慢。因为数据在磁盘上不一定是连续存储的,磁盘读取数据时,需要将磁盘头移动到目标位置,然后读取目标块(页)。这个过程包括寻址时间传输时间,尤其是寻址时间消耗较大。树越高,查询过程中需要访问的磁盘块(页)越多,每次访问都需要一次磁盘 I/O。所以MySQL选择构建索引的数据结构的根本目标就是,降低树的高度

2. 1、二叉树(平衡二叉树)

  二叉树每个节点最多只能有两个子节点。每个节点可以视为一个Node对象,包含自身的值,左子节点,右子节点,父节点。平衡二叉树是对于二叉树的改进,在构造时加入了小于根节点的在左,大于根节点的在右,相同的不存这样的原则。但是在极端情况下,会形成链表
在这里插入图片描述  这样和降低树的高度的目的很显然是互相违背的,并且平衡二叉树,在插入元素时会频繁的发生旋转的情况,对于性能的开销也是比较大的,所以不采用该方案。

2.2、红黑树

  红黑树是针对于平衡二叉树的改进方案,通过变色减少旋转的次数,性能要优于平衡二叉树。但是对于降低树的高度,有后续更好的解决方案。

2.3、Hash表

  Hash 表通过哈希函数将键值映射到特定的存储位置。由于这种映射是离散的,数据没有按顺序排列,导致的一个很大的问题就是只支持单点精确查询,无法做到范围查询。并且Hash 索引依赖完整的键值生成哈希值,只能进行等值匹配,不支持模糊查询。在发生Hash冲突时,也需要浪费一部分性能去进行处理,缺点较多,所以不采用该方案。

2.4、B树

  B树是一种平衡的多路搜索树,其每个节点最多可以有多个子节点,和普通的树相比,每个节点存储多个键值,(平衡二叉树,每个节点只能存储一个键值)并按照从小到大的顺序排列。并且一个节点内的键值将子树划分为不同的范围:

对于一个节点 [K1, K2, K3]:
左子树的所有键 < K1
中间子树的所有键满足K1≤ 键 < K2
右子树的所有键 ≥K3。

  并且所有叶子节点都在同一层,保证了树的高度一致,并且无论是叶子节点,还是其他节点,都存放了数据和索引。一页通常是16K。索引和数据并存,会导致页的数量增多,同样无法控制树的高度,所以不采用该方案。
在这里插入图片描述

2.5、B+树

  B+树是针对B树的改进,在B树的基础上,将所有的数据和索引全都存放在叶子节点上,而非叶子节点,只存放关键性的索引(冗余索引),不存放数据。同样一页16k,相比于B树能存放更多的数据。可以显著降低树的高度,并且B+树的叶子节点之间会构成一个双向链表,有利于范围查询。故最终选择使用B+树作为索引的数据结构
在这里插入图片描述
  至于B树和B+树为何每个节点存储多个键值?因为在读取数据/索引时,会将该页加载到内存,然后再进行查找,查找的时间,`远小于将某一页加载到内存的时间。
  假设一棵 B+ 树的节点扇出为 100,树的高度为 ℎ,则第一层(根节点)可以存储 100 个键值。第二层可以存储100^2个键值,第三层可以存放100 ^3个键值,数据量为 1 百万时,树的高度仅需 3 层。并且每个节点存储多个键值,可以充分利用磁盘页的存储空间,减少磁盘页的浪费,并且上面提到,在插入或者删除数据时,索引的结构可能会发生改变,当节点存储多个键值时,插入和删除操作不会频繁导致节点的分裂和合并。(如果一个节点可以存储 100 个键值,只有在键值数量超过 100 或少于 50 时才需要调整。如果一个节点只能存储 1 个键值,插入或删除可能会频繁触发分裂或合并,增加开销。)


http://www.ppmy.cn/embedded/152404.html

相关文章

【2025软考高级架构师】案例题重点知识——第三部分

33.需求分析总结 需求分析主要是用来分析系统主要做什么,提炼、分析、认真审查获取到的需求,确保所有项目干系人明白其中的含义,同时找出错误、遗漏或者不足的地方。 需求分析的7个方面包括: 1.建立系统边界 2.创建用户界面原型 3.创建数据流图 4.创建数据字典 5.确定…

深入Android架构(从线程到AIDL)_15 应用Android的UI框架02

3、 使用UI线程的MQ(Message Queue) // myView.java // ……… public class myView extends View {// ………Override protected void onDraw(Canvas canvas) {super.onDraw(canvas);// ………// canvas.drawRect(….);invalidate();} } 我们可以透过Message方式来触发UI线程…

PingCAP 连续两年入选 Gartner 云数据库管理系统魔力象限“荣誉提及”

近日&#xff0c;全球 IT 市场研究和咨询公司 Gartner 发布最新报告《Magic Quadrant™ for Cloud Database Management Systems》&#xff08;云数据库管理系统魔力象限&#xff09;&#xff0c;PingCAP 因其企业级开源分布式数据库 TiDB 在全球市场的表现&#xff0c;连续两年…

如何在 Spring Cloud Gateway 中创建全局过滤器、局部过滤器和自定义条件过滤器

Spring Cloud Gateway 是一个功能强大的 API 网关&#xff0c;能够处理 HTTP 请求、响应及路由。通过过滤器机制&#xff0c;您可以在请求和响应过程中进行各种处理操作&#xff0c;如记录日志、身份验证、限流等。Spring Cloud Gateway 提供了三种主要类型的过滤器&#xff1a…

AI赋能跨境电商:魔珐科技3D数字人破解出海痛点

跨境出海进入狂飙时代&#xff0c;AI应用正在深度渗透并重塑着跨境电商产业链的每一个环节&#xff0c;迎来了发展的高光时刻。生成式AI时代的大幕拉开&#xff0c;AI工具快速迭代&#xff0c;为跨境电商行业的突破与飞跃带来了无限可能性。 由于跨境电商业务自身特性鲜明&…

ElasticSearch7.8下载、安装教程和快照恢复

一、Windows安装ElasticSearch7.8 下载地址:ElasticSearch 7.8下载地址 &#xff0c;点击进入下载界面后根据自身的操作系统选择对应的版本进行下载即可 下载完成后可以将压缩包elasticsearch-7.8.0-windows-x86_64.zip进行解压&#xff0c;进入解压文件后可以看到以下文件布局…

30天开发操作系统 第 12 天 -- 定时器

前言 定时器(Timer)对于操作系统非常重要。它在原理上却很简单&#xff0c;只是每隔一段时间(比如0.01秒)就发送一个中断信号给CPU。幸亏有了定时器&#xff0c;CPU才不用辛苦地去计量时间。……如果没有定时器会怎么样呢?让我们想象一下吧。 假如CPU看不到定时器而仍想计量时…

Rabbitmq 业务异常与未手动确认场景及解决方案

消费端消费异常&#xff0c;业务异常 与 未手动确认是不是一个场景&#xff0c;因为执行完业务逻辑&#xff0c;再确认。解决方案就一个&#xff0c;就是重试一定次数&#xff0c;然后加入死信队列。还有就是消费重新放入队列&#xff0c;然后重新投递给其他消费者&#xff0c;…