作者推荐 | 【深入浅出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%

在这里插入图片描述

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

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

相关文章

[DIOR | DIOR-R]旋转目标检测数据集——基于YOLOv8obb,map50已达81.8%

DIOR是一个用于光学遥感图像目标检测的大规模基准数据集。涵盖20个对象类。这20个对象类是飞机、机场、棒球场、篮球场、桥梁、烟囱、水坝、高速公路服务区、高速公路收费站、港口、高尔夫球场、地面田径场、天桥、船舶、体育场、储罐、网球场、火车站、车辆和风磨。 1. DIOR简…

常见の算法链表问题

时间复杂度 1.链表逆序 package class04;import java.util.ArrayList; import java.util.List;public class Code01_ReverseList {public static class Node {public int value;public Node next;public Node(int data) {value data;}}public static class DoubleNode {publi…

Java 字符串 05 练习-遍历字符串和统计字符个数

代码: import java.util.Scanner; public class practice{public static void main(String[] args) {//键盘录入一个字符串,并进行遍历;Scanner input new Scanner(System.in);System.out.println("输入一个字符串:")…

webassembly003 whisper.cpp的main项目-1

参数设置 /home/pdd/le/whisper.cpp-1.5.0/cmake-build-debug/bin/main options:-h, --help [default] show this help message and exit-t N, --threads N [4 ] number of threads to use during computation-p N, --processors …

Android App开发-简单控件(2)——视图基础

2.2 视图基础 本节介绍视图的几种基本概念及其用法,包括如何设置视图的宽度和高度,如何设置视图的外部间距和内部间距,如何设置视图的外部对齐方式和内部对齐方式等等。 2.2.1 设置视图的宽高 手机屏幕是块长方形区域,较短的那…

【星海随笔】unix 启动问题记录.

启动Ubuntu操作系统时,直接进入GRUB状态。 调试时候,曾显示 no bootable device no known filesystem detected 注意: 目前 GRUB 分成 GRUB legacy 和 GRUB 2。版本号是 0.9x 以及之前的版本都称为 GRUB Legacy ,从 1.x 开始的就称…

NODE笔记 2 使用node操作飞书多维表格

前面简单介绍了node与简单的应用,本文通过结合飞书官方文档 使用node对飞书多维表格进行简单的操作(获取token 查询多维表格recordid,删除多行数据,新增数据) 文章目录 前言 前两篇文章对node做了简单的介绍&#xff…

eNSP学习——配置通过STelnet登陆系统

目录 背景 实验内容 实验目的 实验步骤 实验拓扑 详细配置过程 基础配置 配置SSH server 配置SSH client 配置SFTP server 与client 背景 由于Telnet缺少安全的认证方式,而且传输过程采用的是TCP进行明文传输。单纯的提供Telnet服务容易招致主机IP地址欺骗、路…

数据分析 - 图形化解释(后续添加)

图形化解释 作为数据分析师来说一个好的图形,就是自己的数据表达能力 简单文本 只有一两项数据需要分享的时候,简单的文本是最佳的沟通方法 下图的对比可以看出来文字的表达效果会好很多 散点图 散点图在展示两件事的关系时很有用,观察是否存…

【搞懂设计模式】命令模式:从遥控器到编程的妙用!

我们都熟悉电视遥控器,它有许多按钮,每个按钮都有确定的功能。你按下电源键电视就会打开,再按下一次电视就会关闭。编程世界里也有这种模式,这就是我们说的命令模式。 命令模式是一种设计模式,它把一个请求或操作封装…

以梦为码,CodeArts Snap 缩短我与算法的距离

背景 最近一直在体验华为云的 CodeArts Snap,逐渐掌握了使用方法,代码自动生成的准确程度大大提高了。 自从上次跟着 CodeArts Snap 学习用 Python 编程,逐渐喜欢上了 Python。 我还给 CodeArts Snap 起了一个花名: 最佳智能学…

计算机服务器中了halo勒索病毒怎么办,halo勒索病毒解密处理流程

计算机技术的发展与应用为企业的生产生活提供了坚实基础,但同时也为网络安全威胁制造了有利条件。近期,网络上的勒索病毒非常嚣张,给企业的计算机服务器带来严重威胁。近日,云天数据恢复中心接到山东某制造公司的求助,…

Oracle DG环境下的秘钥管理

今天有朋友问到1)DG环境下的秘钥管理需要注意什么,2)秘钥管理对DG的日志同步有影响吗? 对于2)的回答是明确的,没有影响。秘钥的管理和DG的redo log shipping完全是两套机制。在最新版的Oracle Key Vault常…

Qlik Sense : ErrorCode(错误变量)

错误变量 所有错误变量的值在脚本执行之后依然保留。第一个变量 ErrorMode 由用户输入,最后三个变量是 Qlik Sense 的输出(包括脚本中错误的信息)。 使用每个变量的下拉列表可查看每个变量的简短描述和语法。单击语法描述中的变量名称可了解…

Vulnhub靶机:FunBox10

一、介绍 运行环境:Virtualbox 攻击机:kali(10.0.2.15) 靶机:FunBox10(10.0.2.35) 目标:获取靶机root权限和flag 靶机下载地址:https://download.vulnhub.com/funbo…

Effective C++——关于重载赋值运算

令operator返回一个*this的引用 在重载,,*等运算符时&#xff0c;令其返回一个指向this的引用。 class MyClass {int* val; public:MyClass(int i) : val(new int(i)){}MyClass():val(new int(0)){}void print() {cout << *val << endl;}MyClass& operator(co…

LabVIEW电液比例阀测试系统

电液比例阀与普通阀和伺服阀相比&#xff0c;比例阀展现出显著的耐污染和可靠性特点。为了满足这些比例阀的综合性能测试需求&#xff0c;开发了一种基于LabVIEW软件的电液比例阀综合性能试验台。这个系统不仅能够进行比例压力阀、流量阀和方向阀的性能测试&#xff0c;而且通过…

RabbitMQ 笔记一

概览&#xff1a; MQ基本概念 RabbitMQ入门 基本工作模 1.MQ是什么&#xff1f; MQ:Message Queue, 存储消息的中间件&#xff0c;是消息发送过程中的暂存容器&#xff0c;主要用于解决分布式系统进程间的通信。 分布式系统通信的两种方式&#xff1a;直接远程调用、借助第三…

01:云计算底层技术奥秘|虚拟化管理|公有云概述

云计算底层技术奥秘&#xff5c;虚拟化管理&#xff5c;公有云概述 虚拟化平台安装验证虚拟化支持 Linux虚拟机创建虚拟机磁盘虚拟机配置文件创建虚拟机 公有云简介 虚拟化平台安装 虚拟化&#xff1a;是一种技术 就是将不可拆分的实体资源变成可以自由划分的逻辑资源&#xf…

one-stage/two-stage区别

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