作者推荐 | 【深入浅出MySQL】「底层原理」探秘缓冲池的核心奥秘,揭示终极洞察

news2024/10/1 21:24:41

探秘缓冲池的核心奥秘,揭示终极洞察

  • 缓存池BufferPool机制
  • MySQL缓冲池
    • 缓冲池
      • 缓冲池的问题
    • 缓冲池的原理
    • 数据预读
      • 程序的局部性原则(集中读写原理)
        • 时间局部性
        • 空间局部性
      • innodb的数据页
        • 查询InnoDB的数据页
        • InnoDB缓冲池缓存数据页
        • InnoDB缓存数据的淘汰算法
        • 传统的LRU是如何进行缓冲页管理
          • 页已经在缓冲池
            • 链表数据结构
          • 页不在缓冲池
        • MySQL的LRU是如何进行缓冲页管理
          • 预读失效
            • 优化预读失效
            • 分代进行LRU缓存处理
            • 案例分析
          • MySQL缓冲池污染
            • 执行过程如下
            • 优化方案
            • 老生代停留时间窗口
  • 最后总结
    • 有三个比较重要的参数
      • innodb_buffer_pool_size
      • innodb_old_blocks_pct
      • innodb_old_blocks_time

缓存池BufferPool机制

应用系统分层架构:一个优化策略是将最常访问的数据存放在缓存中,以加快数据访问速度,避免频繁地访问数据库。
在这里插入图片描述
操作系统:借助缓冲池机制来优化数据访问,从而避免了反复直接访问磁盘的开销,极大地提升了数据访问的速度。缓冲池通过在内存中临时存储最常访问的数据,将频繁读写的I/O操作转化为对内存中数据的操作,极大地降低了磁盘访问的延迟和系统开销。

MySQL缓冲池

MySQL作为一个存储系统,有着一个关键的优化机制——缓冲池(buffer pool),它极大地提高了数据的访问效率,避免了频繁的磁盘IO操作。通过将常用的数据存储在内存中,MySQL可以快速响应查询请求,减少耗时的磁盘访问。这一优化机制在提升数据库性能方面起到了重要的作用。

在MySQL数据库中我们最常用的引擎就是InnoDB,因此我们采用InnoDB的缓冲池进行分析和介绍。

缓冲池

InfoDB引擎为了优化数据访问并提升速度,系统常常将缓存表数据和索引数据加载到缓冲池中,以避免频繁的磁盘IO操作。这种缓存机制大大减少了磁盘IO的开销,同时加速了数据的读取和写入过程。

缓冲池的问题

凡事都具备两面性,抛开数据易失性不说,访问快速的反面是存储容量小:

  • 缓存访问快,但容量小,数据库存储了200G数据,缓存容量可能只有64G;
  • 内存访问快,但容量小,买一台笔记本磁盘有2T,内存可能只有16G;

因此,在优化数据访问时,只能将最常访问的热门数据放置在最近的位置,以最大程度地减少对磁盘的访问。为了更好的可以实现优化数据库以及对应的缓冲池,我们先去研究一下数据库的底层原理机制。

缓冲池的原理

在进行详细介绍之前,让我们先重点了解一下预读的概念。

数据预读

预读是一种优化策略,它通过提前加载数据到缓冲区内存中,以减少磁盘IO的次数和延迟。当系统预测到将来可能需要某些数据时,它会主动将这些数据从磁盘读取到内存中,并且放置在缓冲区,以备后续的快速访问。

磁盘读写并非按需读取,而是按读取的方式进行。每次至少读取一页数据(假设为4KB)。如果未来需要读取的数据正好在这一页中,就可以避免后续的磁盘IO操作,从而提高数据的访问效率。

程序的局部性原则(集中读写原理)

在数据访问中,一般遵循着“集中读写”的原则。也就是说,当使用某个数据时,很有可能会连续使用其附近的数据。这就是著名的“局部性原理”。

基于这个原理,预先加载数据是一种有效的优化策略,因为它可以减少磁盘IO操作次数,提高数据访问的效率

程序的局部性原理是指在程序中存在着数据和指令的访问局部性的倾向。具体来说,局部性原理包括以下两个方面:
在这里插入图片描述

时间局部性

指程序中某个数据项或指令在一段时间内可能被重复访问。例如,循环结构中的数据和指令在每次迭代中都会被反复访问,因此在一段时间内都具有较高的访问概率。

空间局部性

指程序中相邻的数据项或指令很可能被连续访问。这是因为在程序中,数据和指令通常以连续的内存地址存储,因此当访问一个数据或指令时,其附近的数据或指令很可能会被紧接着访问。

innodb的数据页

默认情况下,innodb.pagesize参数的值为16KB,这是InnoDB的推荐值,通常情况下,16KB的页面大小适用于大多数应用,但对于特定的工作负载和硬件环境,可以进行一些测试和调优以确定最佳的页面大小设置。不过,可以根据实际需求进行调整。

较小的页面大小可以提高磁盘空间的利用效率,但可能会导致更多的磁盘IO操作。而较大的页面大小可以减少IO操作,但会占用更多的内存。

查询InnoDB的数据页
  1. 连接到MySQL服务器。

  2. 运行以下命令登录到MySQL命令行界面或查询工具中:

    SHOW VARIABLES LIKE 'innodb_page_size';
    

    或者

    SELECT @@innodb_page_size;
    
  3. 运行该命令后,会返回当前InnoDB数据页的大小。通常情况下,默认的InnoDB数据页大小为16KB。

注意,innodb_page_size是一个只读变量,它反映了当前InnoDB数据页的大小。如果在编译时进行了自定义设置,那么返回的值可能会不同。在选择合适的innodb.pagesize值时,需要综合考虑数据库的性能需求以及服务器硬件配置。

InnoDB缓冲池缓存数据页

磁盘访问按页读取能够提高性能,因此缓冲池通常也按页缓存数据。这种设计有助于减少磁盘IO操作,并提高数据访问的效率,那么InnoDB是以什么算法,来维护这些缓冲页呢?

InnoDB缓存数据的淘汰算法

最常见的数据页置换算法是LRU(Least Recently Used,最近最不常用使用)算法。LRU算法基于一个简单的原则,即最近最不常用的数据页很可能在未来也不会很频繁使用,因此可以被替换出缓冲池以腾出空间给新的页数据。

注意,尽管像内存缓存(例如memcached)和操作系统中的缓冲池都使用LRU算法来进行页置换管理,但MySQL中的InnoDB存储引擎的页置换策略略有不同。

传统的LRU是如何进行缓冲页管理

最常见的数据页置换策略确实是将新加入缓冲池的页放置在LRU链表的头部,作为最新访问的元素,确保它们最后被淘汰。然而,具体的数据页置换策略可以分为以下两种情况:

页已经在缓冲池

只做“转移"LRU头部的动作,而没有页被淘汰;
在这里插入图片描述
考虑到上图,假设缓冲池的LRU长度为10,并且当前缓存的页为1, 3, 5,…, 40, 7。现在需要访问的数据位于页号为4的页中,由于页号为4的页不在缓冲池中,首先需要将其放入LRU链表的头部,表示最近被访问。
在这里插入图片描述
同时,由于缓冲池已满,需要进行淘汰操作。

链表数据结构

为了减少数据移动的开销,常见的做法是使用链表来实现LRU(Least Recently Used,最近最少使用)算法。

具体地说,LRU缓存通常使用双向链表来维护缓存中页的顺序。当一个页被访问时,它会被移动到链表的头部,表示为最近被访问过的页。当需要淘汰页时,可以从链表的尾部移除最久未被访问的页。

页不在缓冲池

除了做"放入"LRU头部的动作,还要做“淘汰"LRU尾部页的动作,假如,再接下来要访问的数据在页号为50的页中。
在这里插入图片描述
页号为50的页,原来不在缓冲池里,把页号为50的页,放到LRU头部,同时淘汰尾部页号为7的页;

MySQL的LRU是如何进行缓冲页管理

这里有两个问题需要考虑,导致MySQL不直接采用类似memcache等软件的方法:

  • 预读失效:在MySQL中,预读机制的有效性受制于访问模式的复杂性。由于MySQL常用于事务性应用和复杂查询,访问模式往往难以准确预测,从而导致预读策略的准确性下降。因此,在MySQL中完全依赖预读机制无法保证高效的数据访问。

  • 缓冲池污染:缓冲池是MySQL用于存储数据页的关键组件,它存储了最常访问的数据和索引。与简单的缓存系统不同,MySQL的缓冲池需要维护多种复杂的数据结构,如锁、日志等,以保证ACID事务的一致性。

预读失效

在某些情况下,由于预读机制(Read-Ahead)的存在,某些页被提前放入了缓冲池。然而,最终MySQL并没有从这些页中获取所需的数据,这被称为预读失效。

优化预读失效
  • 预读失败的页,停留在缓冲池LRU里的时间尽可能短;
  • 真正被读取的页,才挪到缓冲池LRU的头部,以保证,真正被读取的热数据留在缓冲池里的时间尽可能长。
分代进行LRU缓存处理

将LRU划分为新生代和老生代两个部分,可以更加高效地管理缓冲池中的页。热度高的页往往在新生代中得到缓存,并更长时间地保持在缓冲池中,从而提高其快速访问的可能性。相反,热度较低的页会逐渐被移动到老生代,让出空间给新的热页,并减少对LRU链表的操作次数。

新老生代收尾相连,即:新生代的尾(tail)连接着老生代的头(head)。

  • 新生代:用来缓存最近被访问的页的部分,它通常拥有较小的容量。当一个页被访问时,它会被移动到新生代的头部,表示为最近的访问,这个阶段被称为“热化”(hot phase),即页被频繁访问的阶段。

  • 老生代:用于缓存较长时间未被访问的页,它通常拥有较大的容量。在新生代中停留一段时间后,如果一个页仍然没有被访问,它会被移动到老生代的头部。这个阶段被称为“冷化”(cold phase),即页的热度降低并逐渐被冷落的阶段。
    在这里插入图片描述
    以一个例子来说明,整个缓冲池的LRU可以如上图所示:缓冲池的总长度为10,前7个页是新生代,接下来的3个页是老生代,新生代和老生代首尾相连。

案例分析

在这个例子中,前7个页(页号4至页号6)位于新生代,它们是最近被访问的,因此被放置在LRU链表的头部。接下来的3个页(页号8至页号10)位于老生代,它们是相对较旧的页,在新生代的页都被放满之后才会被放置。
在这里插入图片描述
场景1:页号为50的新页被预读加入缓冲池,当页面50从老年代头部插入时,老年代尾部的页面(整体尾部)将被淘汰。假设页面50不会被真正读取,即预读失败,它将比新生代的数据更早从缓冲池中淘汰出去。

场景2:页号50立即被读取,例如,SQL访问了页面中的行数据,那么页面50将立即被移到新生代的头部,并将新生代的页面挤出,进入老年代。在这种情况下,没有页面被真正淘汰。

改进版的缓冲池LRU算法能够有效解决"预读失败"的问题,不要因为害怕预读失败而取消预读策略,因为大部分情况下,局部性原理是成立的,预读是有效的

MySQL缓冲池污染

当某个SQL语句需要批量扫描大量数据时,可能会导致将缓冲池中的所有页面替换出去,进而导致热数据被移出缓冲池,从而导致MySQL性能急剧下降。这种情况被称为缓冲池污染。

例如,有一个数据量较大的表,当执行之后,虽然结果集可能只有少量数据,但这类like不能命中索引,必须全表扫描,就需要访问大量的页。

执行过程如下
  1. 将页面加载到缓冲池中,并将其插入到老年代的头部。
  2. 从页面中读取相关的行数据,并将其插入到新生代的头部。
  3. 对每个行数据的条件字段与预想值进行比较,如果符合条件,则将其加入到结果集中。
  4. 继续扫描所有页面中的所有行数据,直到完成。

然而,这种方式会导致所有的数据页面都被加载到新生代的头部,但只会访问一次,这将导致真正的热数据被大量换出。

优化方案

在这里插入图片描述

老生代停留时间窗口

假设 T 为老生代停留时间窗口。

  1. 插入到老生代头部的页面,即使立即被访问,也不会立即放入新生代头部。
  2. 只有当页面满足两个条件时,才会被放入新生代头部:被访问过,并且在老生代停留时间大于 T。
    在这里插入图片描述
    这意味着,即使页面被立即访问,也不会立即被移动到新生代头部。只有在页面被访问且在老生代停留时间。

继续举例,假如批量数据扫描,有51,52,53,54等4个数据页将要依次被访问。
在这里插入图片描述
如果没有“老生代停留时间窗口”的策略,这些批量被访问的页面,会换出大量热数据。加入“老生代停留时间窗口”策略后,短时间内被大量加载的页,并不会立刻插入新生代头部,而是优先淘汰那些,短期内仅仅访问了一次的页。而只有在老生代呆的时间足够久,停留时间大于T,才会被插入新生代头部。

最后总结

  • 预读机制:给我们一个启示,即可以将一些可能需要访问的页提前加载到缓冲池中,以避免未来的磁盘IO操作。通过提前加载数据,我们可以利用局部性原理,预测并预先缓存未来可能用到的数据页,从而提高数据访问的性能和效率,减少响应时间。

  • MySQL在设计上需要综合考虑事务性、复杂查询等方面的要求,采用了更加复杂的缓冲池管理方式,以确保高性能和数据一致性。这包括使用LRU算法、预读机制、自适应策略等来最大程度地利用内存资源,同时解决预读失效和缓冲池污染等问题,并提供高效、稳定的数据库服务。

缓冲池污染:由于大量数据扫描操作而引起的缓冲池中的热数据被替换出去的情况,为了解决这个问题,可以通过合理配置缓冲池的大小,调整相关缓存参数,或者改进SQL语句的扫描方式,以减少对缓冲池的影响,从而提高MySQL的性能。

有三个比较重要的参数

innodb_buffer_pool_size

innodb_buffer_pool_size 是 MySQL 中 InnoDB 存储引擎的一个配置参数,用于指定 InnoDB 缓冲池的大小。

mysql>show variables like '%innodb_buffer_pool_size%';

在这里插入图片描述

innodb_old_blocks_pct

innodb_old_blocks_pct是InnoDB存储引擎的一个参数,用于指定LRU链表中被认为是老生代页的比例。

默认情况下,innodb_old_blocks_pct的值为37。这意味着LRU链表中的前37%的页将被视为新生代,而后63%的页将被视为老生代。

mysql>show variables like '%innodb_old_blocks_pct%';

在这里插入图片描述
较大的值表示更多的页被视为老生代页,而较小的值则表示更少的页被认为是老生代页。

innodb_old_blocks_time

innodb_old_blocks_time 的单位是秒 (s), MySQL 中 InnoDB 存储引擎的一个配置参数,用来确定一个数据块在缓冲池中没有被访问的时间超过多久后被认为是"旧"的。

mysql>show variables like '%innodb_old_blocks_time%

在这里插入图片描述

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

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

相关文章

区间dp/线性dp,HDU 4293 Groups

一、题目 1、题目描述 After the regional contest, all the ACMers are walking alone a very long avenue to the dining hall in groups. Groups can vary in size for kinds of reasons, which means, several players could walk together, forming a group.   As the …

RuoYi-Vue前后端分离后台开发框架运行详细教程

一、官网下载代码 RuoYi-Vue是一款基于SpringBootVue的前后端分离极速后台开发框架。 若依官网:http://ruoyi.vip演示地址:http://vue.ruoyi.vip代码下载:https://gitee.com/y_project/RuoYi-Vue 下载之后解压,ruoyi-ui是前端代…

【JavaWeb】过滤器 Filter

文章目录 过滤器是什么?一、过滤器概述二、过滤器工作位置图解三、Filter接口API四、过滤器使用4.1 定义一个过滤器类,编写功能代码:4.2 xml配置:4.3 定义 servletG 目标资源 模拟测试 :4.4 过滤图解 五、过滤器生命周期六、过滤器链的使用6.…

香港服务器IP段4c和8c的区别及SEO选择建议

随着互联网的快速发展,服务器IP段的选择对于网站SEO优化至关重要。香港服务器IP段4C和8C是两种常见的IP段,它们在SEO优化中具有不同的特点和优势。本文将详细介绍这两种IP段的区别,并给出相应的SEO选择建议。 一、香港服务器IP段4C和8C的区别…

每日coding 2846. 边权重均等查询 236. 二叉树的最近公共祖先 35. 搜索插入位置 215. 数组中的第K个最大元素 2. 两数相加

2846. 边权重均等查询 xs,已放弃,考到直接寄 236. 二叉树的最近公共祖先 236. 二叉树的最近公共祖先 给定一个二叉树, 找到该树中两个指定节点的最近公共祖先。 百度百科中最近公共祖先的定义为:“对于有根树 T 的两个节点 p、q&…

向量表示在自然语言、知识图谱和图像视觉中的应用

目录 前言1 不同应用的向量表示1.1 自然语言中的向量表示1.2 知识图谱中的向量表示1.3 图像视觉中的向量表示 2 词的向量表示2.1 One-hot encoding的限制2.2 Bag-of-Words模型的不足2.3 Word Embedding的引入2.4. 词的分布式表示与语境关系 3 词向量模型3.1 CBoW(Co…

swagger2 和 knife4j 整合

swagger整合knife4j 导入依赖 <dependency><groupId>com.github.xiaoymin</groupId><artifactId>knife4j-spring-boot-starter</artifactId><version>3.0.2</version></dependency>引入配置 我们自己写一个配置类也好,我这里写…

乐观锁的底层实现以及如何解决ABA问题

什么是乐观锁&#xff1f;乐观锁底层是如何实现的&#xff1f; 乐观锁是一种并发控制的策略。在操作数据的时候&#xff0c;线程读取数据的时候不会进行加锁&#xff0c;先去查询原值&#xff0c;操作的时候比较原来的值&#xff0c;看一下是都被其他线程修改&#xff0c;如果…

OpenHarmony—TypeScript到ArkTS约束说明

对象的属性名必须是合法的标识符 规则&#xff1a;arkts-identifiers-as-prop-names 级别&#xff1a;错误 在ArkTS中&#xff0c;对象的属性名不能为数字或字符串。通过属性名访问类的属性&#xff0c;通过数值索引访问数组元素。 TypeScript var x { name: x, 2: 3 };c…

STM32标准库开发—W25Q64详细介绍

W25Q64简介 Flash编程原理都是只能将1写为0&#xff0c;而不能将0写成1.所以在Flash编程之前&#xff0c;必须将对应的块擦除&#xff0c;而擦除的过程就是将所有位都写为1的过程&#xff0c;块内的所有字节变为0xFF.因此可以说&#xff0c;编程是将相应位写0的过程&#xff0c…

Ubuntu 22.04 安装tomcat

tomcat是常用的Java服务容器,这篇文章我们就来讲讲如何安装它。 更新软件包 首先是更新软件包,这是最常规的操作 sudo apt update 然后是开始安装,不多一会就可以安装好了 sudo apt install tomcat9 然后看一下状态 sudo systemctl status tomcat9 发现虽然启动了,但…

IS-IS:03 ISIS链路状态数据库

一个 OSPF 链路状态数据库是若干条 LSA 的集合。与此相似&#xff0c;一个 IS-IS 链路状态数据库是若干条 LSP 的集合。与 OSPF 链路状态数据库不同&#xff0c; IS-IS 链路状态数据库有 level-1 和 level-2 之分。 在IS-IS 协议中&#xff0c;每一条 LSP 都有一个剩余生存时间…

自学Java的第48,49,50,51天

IO流 应用场景 IO流的分类 文件字节输入流 写法 读取一个字节 读取多个字节 优化&#xff1a; 注意&#xff1a; 读取全部字节 写法 注意&#xff1a; 文件字节输出流 写法 案例&#xff1a; 写法 释放资源的方法 try-catch-finally 写法 try-with-resource 写法 字符流 …

linux内网搭建NFS网络文件系统(rpm)

linux 内网搭建nfs网络文件系统&#xff08;rpm包&#xff09; 前言&#xff1a;一、上传安装包到服务器二、NFS服务端配置三、建立共享目录(服务器端和客户端)四、添加配置共享目录&#xff08;服务器端&#xff09;五、NFS客户端配置六、测试共享服务 前言&#xff1a; 用自…

Type-C平板接口协议芯片介绍,实现单C口充放电功能

在现代平板电脑中&#xff0c;Type-C接口已经成为了一个非常常见的接口类型。相比于传统的USB接口&#xff0c;Type-C接口具有更小的体积、更快的传输速度和更方便的插拔体验。但是&#xff0c;在使用Type-C接口的平板电脑上&#xff0c;如何实现单C口充电、放电和USB2.0数据传…

【iOS ARKit】同时开启前后摄像头BlendShapes

在上一节中已经了解了 iOS ARkit 进行BlendShapes的基本操作&#xff0c;这一小节继续实践同时开启前后摄像头进行人脸捕捉和世界追踪。 iOS设备配备了前后两个摄像头&#xff0c;在运行AR 应用时&#xff0c;需要选择使用哪个摄像头作为图像输人。最常见的AR 体验使用设备后置…

修复WordPress内部服务器错误的步骤及解决方案

WordPress是一款广泛使用的开源内容管理系统&#xff0c;但在使用过程中&#xff0c;可能会遇到各种内部服务器错误。这些错误可能由于多种原因引起&#xff0c;例如插件冲突、文件权限问题、服务器配置不当等。为了帮助您快速解决这些问题&#xff0c;本文将为您提供一套详细的…

行测-言语:2.语句表达

行测-言语&#xff1a;2.语句表达 1. 语句排序题 捆绑就是看两句话是不是讲的同一个内容&#xff0c;相同内容的句子应该相连。 1.1 确定首句 1.1.1 下定义&#xff08;……就是 / 是指&#xff09; A 1.1.2 背景引入&#xff08;随着、近年来、在……大背景 / 环境下&#…

五招搞定找不到vcruntime140.dll无法继续执行此代码问题

在计算机系统或应用程序运行过程中&#xff0c;如果出现“找不到vcruntime140.dll”这一错误提示&#xff0c;可能会引发一系列的问题和影响。vcruntime140.dll是Microsoft Visual C Redistributable的一部分&#xff0c;这是一个至关重要的运行库文件&#xff0c;对于许多基于…

one-stage/two-stage区别

One-stage和Two-stage是目标检测中的两种主要方法&#xff0c;它们在处理速度和准确性上存在显著差异。以下是两者的主要区别&#xff1a; 处理流程&#xff1a;One-stage方法通过卷积神经网络直接提取特征&#xff0c;并预测目标的分类与定位&#xff0c;一步到位&#xff0c…