为什么MySQL索引选择B+树而不使用B树?

news2025/1/17 5:48:26

为什么mysql索引选择B+树而不使用B树?


1. 关于mysql查询效率:

在这里插入图片描述


2. 关于分块读取:

在这里插入图片描述


3. 关于数据格式存储:

在这里插入图片描述


4. 关于合适的数据结构:哈希表,树

哈希表:

在这里插入图片描述
分析:

    1. 哈希表是散列表,存储在其中的数据是无序的, 所以当进行范围查询的时候,需要挨个便利,效率较低;
    1. 存储过程中会出现哈希碰撞,哈希冲突,必须要设计应能优良的哈希算法;
    1. 存储的过程中,可能会造成存储空间的浪费;
    1. mysql的 memory 存储引擎是 哈希索引;mysql的 innodb 中有一个特性叫做 自适应hash;

在这里插入图片描述

数据结构在线动态展示:https://www.cs.usfca.edu/~galles/visualization/Algorithms.html

如:一个二叉树:单个节点只存储一条数据,单个节点至多有两个子节点。
在这里插入图片描述
随着数据量的增加,树的层数自然也要增加,而树的层数的增加会导致io次数的增加(每层一次io),为了存储更多的数据而不增加io次数,引入了多叉树,即B树

B树

如下:我们选择了一个度为4(每个节点至多存储3条数据)的B树,同样的数据量的存储,我们的树的层数由原本的4层,变成了2层,减少了io查询次数。
在这里插入图片描述
那么,随之而来的又有一个问题,三层的b树能存储多少条数据呐?

B树的存储结构:每个节点可存储多条数据,且存储了对应的键值,指针和数据。

在这里插入图片描述
分析:
innodb默认页的大小是16k,每个树的节点除了存储数据之外,还包含键值指针,我们假设理想模式下,一个数据是1k,那么第一层最多最多存储16条数据;第二层,就是第一层的16个指针对应第二层的16个节点,我们也假设每个节点最多存储16条,那么第二层至多存储数据也就是16 * 16条;第三层,就是第二层的16 * 16个指针,我们也假设每个节点最多存储16条,那么第三层至多存储就是16 * 16 * 16条数据,综上,三层B树最多能存储的数据就是 16 + 16 * 16 + 16 * 16 * 16 = 4368,而实际上一定是小于这个数的,所以三层的B树至多也就存储4000多条数据

4000条数据显然不满足我们平时数据库的存储使用,那么怎么才能存储更多的数据呐?B树继续加层数?多一层就多一次io,代价比较高,不适合;那么,还有另外一种方案,就是每个节点存储尽可能多的数据,因此引入了B+树

B+树

如下:度为4(每个节点至多存储3条数据)的B+树,同样数据量的存储
在这里插入图片描述
B+树的存储结构:上层节点只存储指针和键值,最底层叶子节点存储键值和数据,并且叶子节点之间是链式环结构。与B树相比,上层节点可以存储更多的数据,且叶子节点的范围查询或分页查询效率更高。

在这里插入图片描述
那么,三层的B+树能存储多少条数据呐?

分析:

指针和键值的大小肯定远小于数据的大小,假设指针+键值占用10Byte,innodb默认页的大小是16k,那么第一层磁盘块能存储的指针+键值就是16 * 1024 / 10 = 1600(约等于),假设第一层1600个 指针 指向第二层1600个子节点,第二层的每个子节点一样至多存储指针+键值16 * 1024 / 10 = 1600(约等于),那么第二层总共存储的指针+键值总数就是 1600 * 1600,第三层假设只存储数据,每条数据1k,每个节点至多可以存储16条数据,第二层的1600 * 1600个指针对应第三层的1600 * 1600个子节点,那合起来就是 1600 * 1600 * 16 = 4096 0000条数据;

不同的数据类型,肯定会影响存储的数据量,一般结论是:

3层B+树大概可以存:
主键为bigint:约2000w
主键为int:约4000w

因此,与B树相比,B+树显然可以存储更多的数据量


5. 问题总结:

    1. 一般innodb索引层数有几层?
      解析:
      一般情况下,3-4层,因为3-4层的B+树足以支撑千万级别的数据存储;
    1. 索引列的键(key)值怎么选?
      解析:
      innodb非叶子节点的存储是 指针+键值,指针一般变化不大,所以索引列要尽可能选择占用空间小的字段,因为占用越小,单个节点能存储的 指针+键值 自然就更多,存储的数据自然就更多。
    1. 创建表的时候,主键是否需要自增?
      解析:
      需要,因为如果不自增,会带来索引维护成本的提高,造成叶分裂,而自增只需要在后面有序排列,因此,在满足业务场景的前提下,能自增尽量设置成自增;
    1. 一张表可以有多少个索引?
      解析:
      理论上来说没有索引个数的限制,但并不是索引越多越好。
    1. 一个索引对应一棵树还是多个索引对应多棵树?
      解析:
      一个索引一棵树
    1. 索引的叶子节点要存放数据,当多个索引存在的时候,数据放几份?
      解析:
      一份
    1. 如果数据存储一份,其他索引的叶子节点存储什么数据?
      解析:
      官方文档写的是 primary key,翻译过来的话是 主键,但是回答 聚簇索引的列值 更为准确;
      因为,在innodb的存储引擎中,mysql在数据插入的时候,必须要选择某一个索引列绑定在一起,如果有主键,选择主键,如果没有主键,就会选择唯一键,如果没有唯一键,那么mysql会自动生成一个6字节的rowId来进行存储。

聚簇索引:跟数据绑定存储的称之为聚簇索引;
非聚簇索引:没有跟数据绑定存储的称之为非聚簇索引;
如:
一个表中有 id,name, age, gender, address等字段,其中id为主键,name为普通索引;
那么,id就是聚簇索引,name就是非聚簇索引。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1064838.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

基于腾讯云的OTA远程升级

一、OTA OTA即over the air,是一种远程固件升级技术,它允许在设备已经部署在现场运行时通过网络远程更新其固件或软件。OTA技术有许多优点,比如我们手机系统有个地方做了优化,使用OTA技术我们就不用召回每部手机,直接通过云端就可…

vim操作教程,看这一篇绝对足够啦~

简介 vi 是一种命令行的文本编辑器,vim 是进阶版的 vi,gvim是vim的图形化版本。 Vim有几种基本的工作模式: 一般模式(Normal Mode, 命令模式,普通模式) 一般模式是vim默认的模式,当打开一个…

基于springboot的论坛网站

目录 前言 一、技术栈 二、系统功能介绍 用户信息管理 普通管理员管理 交流论坛 交流论坛评论 三、核心代码 1、登录模块 2、文件上传模块 3、代码封装 前言 随着信息技术在管理上越来越深入而广泛的应用,管理信息系统的实施在技术上已逐步成熟。本文介绍了…

前端项目nginx部署

进入nginx下载地址:https://nginx.org/ 下载完安装包以后,解压在D盘中 双击进去> 将前端打包好的文件放在nginx的html文件夹中 可能80端口会被系统所占用 我们可以在nginx的conf文件夹中的nginx.conf文件中修改80为90 之后我们就可以在任务管理器中看到了 然后 localhost:…

C++入门-day02

引言:在上一节中我们接触了C中的命名空间,学会了C中的标准输出流。这一节,我标题一们讲讲缺省、重载。 一、缺省参数 在C中,给函数的形参默认给一个值就是缺省参数,你可能会比较懵逼,下面看一段代码。 正常…

Prompt-Tuning(一)

一、预训练语言模型的发展过程 第一阶段的模型主要是基于自监督学习的训练目标,其中常见的目标包括掩码语言模型(MLM)和下一句预测(NSP)。这些模型采用了Transformer架构,并遵循了Pre-training和Fine-tuni…

项目规划得心应手:Plane 助你打造高效能团队 | 开源日报 No.48

streamlit/streamlit Stars: 27.5k License: Apache-2.0 Streamlit 是一个快速构建和共享数据应用程序的方法。它可以将数据脚本转换为可分享的 Web 应用,只需几分钟即可完成。该项目完全由 Python 编写,开源且免费!一旦创建了一个应用程序&…

用IDEA操作数据库--MySQL

IDEA集成了DataGrip的操作数据库的功能 就可以省略我们下载SQLyog/Navicat/DataGrip这些图形化操作工具了 以下是IDEA的使用 输入数据库的用户和密码

c++系列之string的模拟实现

💗 💗 博客:小怡同学 💗 💗 个人简介:编程小萌新 💗 💗 如果博客对大家有用的话,请点赞关注再收藏 🌞 string() //注意事项: 1.初始化列表随声明的顺序进行初始化 2.cons…

nodejs+vue晨拾酒馆管理系统elementui

晨拾酒馆管理系统,主要的模块包括管理员;系统首页、个人中心、用户管理、图书分类管理、图书信息管理、图书借阅管理、图书归还管理、图书入库管理、热门图书管理、论坛管理、系统管理,用户;系统首页、个人中心、图书借阅管理、图…

TomCat关键技术

一、Tomcat 是什么 Tomcat 是一个 HTTP 服务器。通过前面的学习,我们知道HTTP 协议就是 HTTP 客户端和 HTTP 服务器之间的交互数据的格式,同时也通过 ajax 和 Java Socket 分别构造了 HTTP 客户端。HTTP 服务器我们也同样可以通过 Java Socket 来实现. 而 Tomcat 就是基于 J…

.net framework中webapi使用swagger进行接口文档展示

第一步:在nuget程序包管理中搜索“Swashbuckle”包,然后进行安装(注:如果是.net core api请安装Sawshbuckle aspnetcore)。 第二步:打开项目App_Start文件夹,修改SwaggerConfig.cs配置文件 我这…

CSS3实现动画加载效果

<!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /><meta name"viewport" content"widthdevice-width, initial-scale1.0" /><title>加载效果</title><link rel"style…

R语言通过接口获取网上数据平台的免费数据

大家好&#xff0c;我是带我去滑雪&#xff01; 作为一名统计学专业的学生&#xff0c;时常和数据打交道&#xff0c;我深知数据的重要性。数据是实证研究的重要基础&#xff0c;每当在完成一篇科研论文中的实证研究部分时&#xff0c;我都能深刻体会实证研究最复杂、最耗时的工…

ubuntu配置vscode c++环境

下载vscode deb安装包 Get Started with C on Linux in Visual Studio Code 1. 安装vscode sudo dpkg -i code_1.83.0-1696350811_amd64.deb 2. 确保gcc编译器已经安装 g --version 如果没有安装&#xff0c;执行以下命令安装 sudo apt-get update sudo apt-get install …

数字IC前端学习笔记:数字乘法器的优化设计(华莱士树乘法器)

相关阅读 数字IC前端https://blog.csdn.net/weixin_45791458/category_12173698.html?spm1001.2014.3001.5482 进位保留乘法器依旧保留着阵列的排列规则&#xff0c;只是进位是沿斜下角&#xff0c;如果能使用树形结构来规划这些进位保留加法器&#xff0c;就能获得更短的关键…

Postman接口测试学习之常用断言

什么是断言&#xff1f; 断言——就是结果中的特定属性或值与预期做对比&#xff0c;如果一致&#xff0c;则用例通过&#xff0c;如果不一致&#xff0c;断言失败&#xff0c;用例失败。断言&#xff0c;是一个完整测试用例所不可或缺的一部分&#xff0c;没有断言的测试用例…

linux虚拟机查看防火墙状态

linux虚拟机查看防火墙状态 在Linux虚拟机中&#xff0c;你可以通过以下几种方法查看防火墙状态&#xff1a; 查看iptables防火墙状态 对于使用iptables防火墙的Linux系统&#xff0c;可以使用以下命令查看防火墙状态&#xff1a; sudo iptables -L -v -n查看firewalld防火墙…

javaee ssm框架项目整合thymeleaf 项目结构图

搭建ssm框架项目 参考这篇博客 引入thymeleaf 引入jar包 <?xml version"1.0" encoding"UTF-8"?><project xmlns"http://maven.apache.org/POM/4.0.0" xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schema…