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

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

  • 缓存池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%

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

可解释性AI

方向一:可解释性AI的定义与重要性 1. 什么是可解释人工智能 可解释人工智能(Explainable Artificial Intelligence, XAI)是指智能体以一种可解释、可理解、人机互动的方式,与人工智能系统的使用者、受影响者、决策者、开发者等&…

牛客网---------[USACO 2016 Jan S]Angry Cows

题目描述 Bessie the cow has designed what she thinks will be the next big hit video game: "Angry Cows". The premise, which she believes is completely original, is that the player shoots cows with a slingshot into a one-dimensional scene consistin…

中仕教育:事业单位考试考什么?

事业单位考试分为两个阶段,分别是笔试和面试,考试科目包括公共科目和专业科目两部分。 公共科目内容是公共基础知识、职业能力测试或申论。一种形式为:公共基础知识职业能力测试或职业能力测试申论。另一种形式为:公共基础申论。…

图像字幕中一些广泛使用的技术

文章目录 R-CNNsRNNsLSTMs and GRUsResNet R-CNNs 在图像识别领域,卷积神经网络(CNN)不仅可以识别出图像中的物体,还能检测出这些物体的边界框。如果我们使用传统的CNN进行对象检测,一种方法是在图像上覆盖一层栅格&a…

理想架构的Doherty功率放大器理论与仿真

Doherty理论—理想架构的Doherty功率放大器理论与仿真 参考: 三路Doherty设计 01 射频基础知识–基础概念 ADS仿真工程文件链接:理想架构的Doherty功率放大器理论与仿真 目录 Doherty理论---理想架构的Doherty功率放大器理论与仿真0、Doherty架构的作用…

试卷扫描转化word的功能有吗?分享4款工具!

试卷扫描转化word的功能有吗?分享4款工具! 随着科技的飞速发展,将试卷扫描并转化为Word文档已经成为我们日常学习和工作的常规需求。但是,市面上的扫描工具众多,如何选择一个既方便又准确的工具呢?本文将为…

JDWP原理分析与漏洞利用

JDWP(Java DEbugger Wire Protocol):即Java调试线协议,是一个为Java调试而设计的通讯交互协议,它定义了调试器和被调试程序之间传递的信息的格式。说白了就是JVM或者类JVM的虚拟机都支持一种协议,通过该协议,Debugger 端可以和 target VM 通信,可以获取目标 VM的包括类…

力扣刷题 第十二 边权重均等查询

现有一棵由 n 个节点组成的无向树,节点按从 0 到 n - 1 编号。给你一个整数 n 和一个长度为 n - 1 的二维整数数组 edges ,其中 edges[i] [ui, vi, wi] 表示树中存在一条位于节点 ui 和节点 vi 之间、权重为 wi 的边。 另给你一个长度为 m 的二维整数数…

windows根据pid查看端口号

一.什么是PID 任务管理器中的PID指的是进程标识符(Process Identifier),它用于在操作系统中唯一标识一个进程二.查看JAVA程序的PID jps命令即可三.根据PID查看端口 netstat -ano|findstr pid

QT5.14.2开发的Mysql8.0系统安装部署过程

最近在Windows 11 64位系统下使用QT5.14.2开发了套系统、使用了MYSQL8.0数据库,项目使用mingw-64编译器进行编译,编译完成后使用windeployqt进行发布,并制作安装包,拷贝到工控机Windows10 64位系统上进行安装运行。本文记录下安装…

机器人3D视觉引导半导体塑封上下料

半导体塑封上下料是封装工艺中的重要环节,直接影响到产品的质量和性能。而3D视觉引导技术的引入,使得这一过程更加高效、精准。它不仅提升了生产效率,减少了人工操作的误差,还为半导体封装技术的智能化升级奠定了坚实的基础。 传统…

兄弟HL-2260黑白激光打印机加粉清零方法

兄弟HL-2260是打印机厂商Brother生产的一款黑白激光多功能一体机,兄弟HL-2260打印机是一款高性价比的打印机,广泛应用于办公和家庭的打印设备,它的打印速度快,而且打印效果好。但是,使用久了之后难免会遇到一些问题&am…

Python学习07—字符串类型及操作

一、字符串类型的表示 字符串是由0个或多个字符组成的有序序列。字符串可由一对单引号或一对双引号表示。由于字符串是有序序列,因此,可以对其中的字符进行索引,且在索引时,字符是从0开始编号。 字符串有两类和四种表示方式&…

Linux——进程间通信(共享内存)

目录 system V共享内存 ​编辑 共享内存函数 共享内存的建立过程 shmget函数 shmctl函数 shmat函数 shmdt函数 实例代码 共享内存的特点 system V共享内存 共享内存区是最快的IPC形式。一旦这样的内存映射到共享它的进程的地址空间(即内存通过某种映射关…

Modern C++ std::bind的实现原理-举例

上节《Modern C std::bind的实现原理-CSDN博客》主要讲的是原理&#xff0c;本节举个例子并画图帮助理解。 上程序&#xff1a; #include <functional> #include <iostream>// A function taking three arguments void printValues(int a, double b, const std::…

探秘Dmail:Web3世界的通讯引领者

摘要&#xff1a;在一个充满潜力并且对创新要求严格的领域中&#xff0c;Dmail作为一种开创性的Web3通讯协议应运而生。 1月24日&#xff0c;OKX Jumpstart宣布上线Dmail&#xff0c;在Web3领域引起了巨大反响&#xff0c;这是一个旨在重新定义数字通讯范式的富有远见的项目&a…

【快影】怎么去除视频上不想要的东西?

您好&#xff0c;您可以试试编辑器中的「一键修复」功能&#xff0c;选中主轨或者画中画&#xff0c;点击底部一键修复功能&#xff0c;框选视频中不想要的内容&#xff0c;点击生成&#xff0c;即可预览修复效果&#xff1b;目前该功能是VIP专属功能&#xff0c;开通快影VIP即…

web前端项目-动画特效【附源码】

文章目录 一&#xff1a;赛车游戏动画HTML源码&#xff1a;JS源码&#xff1a;CSS源码&#xff1a;&#xff08;1&#xff09;normalize.css&#xff08;2&#xff09;style.css 二&#xff1a;吉普车动画演示HTML源码&#xff1a;CSS源码&#xff1a;&#xff08;1&#xff09…

Docker 容器内运行 mysqldump 命令来导出 MySQL 数据库,自动化备份

备份容器数据库命令&#xff1a; docker exec 容器名称或ID mysqldump -u用户名 -p密码 数据库名称 > 导出文件.sql请替换以下占位符&#xff1a; 容器名称或ID&#xff1a;您的 MySQL 容器的名称或ID。用户名&#xff1a;您的 MySQL 用户名。密码&#xff1a;您的 MySQL …

【自动化测试】测试开发工具大合集

收集和整理各种测试工具&#xff0c;自动化测试工具&#xff0c;自动化测试框架&#xff0c;觉得有帮助记得三连一下。 欢迎提交各类测试工具到本博客。 通用测试框架 JUnit: 最著名的xUnit类的单元测试框架&#xff0c;但是不仅仅可以做单元测试。TestNG: 更强大的Java测试框…