【MySQL】InnoDB引擎 —— 架构 —— 内存结构

一、概述

MySQL5.5 版本开始,默认使用InnoDB存储引擎,它擅长事务处理,具有崩溃恢复特性,在日常开发中InnoDB存储引擎使用非常广泛。下面是InnoDB架构图,左侧为内存结构,右侧为磁盘结构。

image-20240720222831378

二、内存结构

image-20240720222858706

在左侧的内存结构中,主要分为这么四大块儿: Buffer Pool(缓冲池)、Change Buffer(更改缓冲区)、Adaptive Hash Index(自适应哈希索引)、Log Buffer(日志缓冲区)。 接下来介绍一下这四个部分。

1). Buffer Pool

InnoDB存储引擎基于磁盘文件存储,访问物理硬盘和在内存中进行访问,速度相差很大,为了尽可能 弥补这两者之间的I/O效率的差值,就需要把经常使用的数据加载到缓冲池中,避免每次访问都进行磁盘I/O。

在InnoDB的缓冲池中不仅缓存了索引页和数据页,还包含了undo页、插入缓存、自适应哈希索引以及 InnoDB的锁信息等等。

缓冲池 Buffer Pool,是主内存中的一个区域,里面可以缓存磁盘上经常操作的真实数据,在执行增删改查操作时,先操作缓冲池中的数据(若缓冲池没有数据,则从磁盘加载并缓存),然后再以一定频率刷新到磁盘,从而减少磁盘IO,加快处理速度。

缓冲池以Page页为单位,底层采用链表数据结构管理Page。根据状态,将Page分为三种类型:

从图中我们也可以看见,每一个块的颜色不同,不同的颜色实际上代表的是页的状态不同。

  • free page:空闲page,申请了这块空间,但是未被使用。
  • clean page:被使用page,数据没有被修改过,还是干净的。
  • dirty page:脏页,被使用page,数据被修改过,也中数据与磁盘的数据产生了不一致。

在专用服务器上,通常将多达80%的物理内存分配给缓冲池 。参数设置: show variables like ‘innodb_buffer_pool_size’;

image-20240720223005100


2). Change Buffer

Change Buffer,更改缓冲区(针对于非唯一二级索引页,也就是对于唯一索引,或者主键索引是不会操作更改缓冲区的),在执行DML语句时,如果这些数据Page 没有在Buffer Pool中,不会直接操作磁盘,而会将数据变更存在更改缓冲区 Change Buffer 中,在未来数据被读取时,再将数据合并恢复到Buffer Pool中,再将合并后的数据刷新到磁盘中。 Change Buffer的意义是什么呢?

先来看一幅图,这个是二级索引的结构图:

image-20240720223032198

与聚集索引不同,二级索引通常是非唯一的,并且以相对随机的顺序插入二级索引。对于聚集索引,像主键索引,通常来说都是我们按照主键的顺序去插入的,顺序插入它就会顺序的进行磁盘IO。

同样,删除和更新 可能会影响索引树中不相邻的二级索引页,比如我们在删除的时候有可能一会删的是这个数据,一会删的又是那个数据;在插入的时候我们有可能一会插入的是这一块的数据,那么每一次操作都是随机磁盘IO。

如果每一次都操作磁盘,会造成大量的磁盘IO。有了 ChangeBuffer之后,我们不用每一次都直接去操作磁盘IO,在这一块操作的时候,可以先去操作ChangeBuffer,然后再以一定的频率把ChangeBuffer中的数据同步到Buffer Pool,然后再刷新到磁盘中,我们可以在缓冲池中进行合并处理,减少磁盘IO。


3). Adaptive Hash Index

自适应hash索引,用于优化对Buffer Pool数据的查询。MySQL的innoDB引擎中虽然没有直接支持 hash索引,它支持的是B+树索引,但是给我们提供了一个功能就是这个自适应hash索引。因为前面我们讲到过,hash索引在 进行等值匹配时,一般性能是要高于B+树的,因为hash索引一般只需要一次IO即可(前提是不存在hash冲突的情况下),而B+树,可能需要几次匹配,所以hash索引的效率要高,但是hash索引又不适合做范围查询、模糊匹配等。

所以在InnoDB中就给我们提供了这样的一个功能:自适应hash,用于优化BUffer Pool中的数据查询。

InnoDB存储引擎会监控对表上各索引页的查询,如果观察到在特定的条件下hash索引可以提升速度, 则建立hash索引,称之为自适应hash索引。

自适应哈希索引,无需人工干预,是系统根据情况自动完成。

参数: adaptive_hash_index,自适应hash索引的开关。

我们简单看一下这个参数默认到底开了没有,如下图,可以发现是开启的。

image-20240721165608317

4). Log Buffer

Log Buffer:日志缓冲区,用来保存要写入到磁盘中的log日志数据(redo logundo log), 默认大小为 16MB,日志缓冲区的日志会定期刷新到磁盘中。也就是说我们在记录日志的时候,为了提高效率,我们可以先把日志记录在缓冲区中,然后定期的刷新到磁盘中。

如果需要更新、插入或删除许多行的事务,增加日志缓冲区的大小可以节省磁盘 I/O。

参数:

innodb_log_buffer_size:缓冲区大小

image-20240721165754655

innodb_flush_log_at_trx_commit:日志刷新到磁盘时机,取值主要包含以下三个:

1:日志在每次事务提交时写入并刷新到磁盘,默认值。

0:每秒将日志写入并刷新到磁盘一次。

2: 日志在每次事务提交后写入,并每秒刷新到磁盘一次。

image-20240720223140573

对于MySQL服务器来说,如果我们MySQL是有一台专门的服务器,那么这台服务器通常来说,80%的内存可能都会分配给它的缓冲区,以此来提高MySQL的执行效率。

  • 4
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值