深入了解Redis——持久化

news2024/11/23 4:09:51

一,Redis持久化

Redis持久化即将内存中的数据持久化到磁盘中,在下一次重启后还能进行使用,Redis持久化分为RDBAOF两种,我们接下来分别介绍RDB和AOF的内部原理和区别

RDB

Redis运行时会将当前的内存快照存入至磁盘中,Redis重新启动后会将快照以二进制的形式给加载进入内存中,其中rdbSaverdbLoad函数至关重要

image

什么时候会触发RDB?

  • Redis停机时会触发
  • 使用命令 save m n
  • 主从同步执行全量复制
  • debug reload命令重新加载Redis

保存

在RDB文件期间,主进程会被阻塞,直至保存完成,其中也分为了两种不同的保存方式SAVEBGSAVE

def SAVE():
 	rdbSave()
 
def BGSAVE():
     pid = fork()
     if pid == 0:
    	 # 子进程保存 RDB
     	rdbSave()
     elif pid > 0:
     	handle_request()
 	 else:
         # pid ==-1
         # 处理 fork 错误
        handle_fork_error()
  • SAVE: 该方法会阻塞Redis主进程,此时不会响应用户请求,直至保存完成位置

image

  • **BGSAVE:**该方法不会阻塞Redis主进程,主线程会查看是否已经fork了一个子线程,如果fork则返回,否则fork一个子进程采用CopyOnWrite机制将当前的快照存入磁盘中

image

SAVE,BGSAVE,BGREWRITEAOF能同时执行吗

  • 对于SAVE来说:

    由于SAVE是阻塞当前主进程的,所以此时此刻不管是用户命令还是BGSAVEBGREWRITEAOF都无法执行,在执行SAVE指令前会检查BGSAVE是否在执行,如果在执行则不能进行SAVE

  • 对于BGSAVE来说:

    **BGSAVE与BGSAVE:**通过上述代码我们可以知道,当BGSAVE正在执行时,会检查子进程是否fork如果fork了子进程则直接返回,所以是不能同时执行的

    **BGSAVE与BGREWRITEAOF:**BGSAVE正在执行时,BGREWRITEAOF会延迟收到指令直至BGSAVE执行完毕。如果BGREWRITEAOF正在执行,BGSAVE会直接返回报错,无法同时执行。

载入

Redis服务器启动时,就会进行rdbLoad函数,此时在载入期间每载入1000条数据就会处理一次当前的用户命令,当然这里的用户命令只能时订阅与发布功能相关的,其他的命令都会统一拒绝。

在载入的时候会优先选择AOF,如果没有设置AOF才会使用RDB

RDB文件结构

一个RDB的文件结构如下所示:

image

  • **REDIS:**该字符标识着RDB文件的开始,相当于魔数
  • RDB-VERSION(四字节): 记录了当前文件RDB的版本号,读取的时候要使用对于版本号的方法读取
  • DB-DATA: 该部分会在RDB文件中出现多次,保存着一个服务器上非空数据库的所有数据
  • SELECT-DB: 代表着该键值对所属的数据库号码,读入RDB文件时,会根据该号码不断切换数据库
  • KEY-VALUE-PAIRS: 代表着一个键值对的数据,每个键值对的数据会用以下结构来进行保存

image

OPTIONAL-EXPIRE-TIME: 这个代表着当前的键值对过期时间,如果没有则为null

TYPE-OF-VALUE: 代表着该键值对以什么样的类型进行存储,会根据不同的类型来进行VALUE的读取(这个地方内容较多,暂不介绍)

KEY: 存储着当前的键

VALUE: 存储着当前键保存的值

  • EOF: 标志数据库内容的结尾,并不是文件的末尾
  • CHECK-SUM: 文件内容校验和,读取时会对其进行文件内容的校验,如果为0则代表关闭了校验和功能

AOF

AOF以协议文本的方式,将所有对数据库写入的命令记录至AOF文件中,以此达到记录数据库状态的目的

image

AOF运行阶段

同步命令至AOF文件分为三个阶段:

命令传播: 将当前的Redis执行完的命令,以命令请求,命令参数,命令参数个数的形式传输给AOF程序

image

缓存追加: 将命令数据接收,并转换为网络通讯的协议方式,将内容追加至AOF缓存当中。

文件写入和保存: 将缓存的内容根据设定的AOF条件写入至AOF文件末尾,此时会调用fsync函数或者fdatasync函数来将写入内容保存至磁盘中

此时会调用aof.c/flushAppendOnlyFile函数来执行以下两个工作:

WRITE(主进程阻塞):根据写入条件,将当前aofBuf的内容写入至AOF文件末尾,这个是写入至文件缓冲区的,写入之后直接返回,如果此时发生宕机,此时写入的内容将丢失

SAVE(主进程看情况阻塞) :根据保存条件,将当前的AOF文件缓存内容保存至磁盘中。采用fsync或fdatasync。

fysnc和fdatasync

参考文献

**fsync:**他会刷新文件的所有修改的核心数据包括文件关联的元数据,再刷新至磁盘中时,他会一直阻塞直至刷新完成

**fdatasync:**他和fsync类似,但是他不会刷新所有的文件元数据,会根据需要来进行刷新

AOF保存模式

Redis 目前支持三种 AOF 保存模式,它们分别是:

AOF_FSYNC_NO :不保存

在不保存的情况下,整个Redis执行期间WRITE会被执行但是不会执行SAVE命令,只有以下几种可能会执行SAVE命令

  • Redis被关闭
  • AOF被关闭
  • 系统的写缓存被刷

这三种情况下的SAVE都会导致主进程阻塞

AOF_FSYNC_EVERYSEC :每一秒钟保存一次

SAVE在原则上会一秒钟执行一次,且这个SAVE是由fork出来的子进程进行执行的,但是值得注意的是这个是原则上面的一秒钟,它是否是每次一秒调用和当前Redis所处的状态有关。

image
  • 子线程正在执行SAVE:

​ 如果执行SAVE时间小于2s:无需进行额外的write和save,程序执行返回

​ 如果执行SAVE时间超过2s:程序执行追加write,但不执行新的save。此时的write必须等待save执行完毕才能进行,所以主线程也会阻塞

  • 子线程没有执行SAVE:

​ 如果上次执行SAVE距今不超过1s:程序执行write但不执行save

​ 如果上次执行SAVE时间距今超过1s:程序执行write和save

所以我们如果在上图的情况1宕机,那此时只会损失小于2s的数据,但是如果在情况发生宕机,此时write已经有2s没写入文件缓存并刷入磁盘,就会有2s的数据损失。所以说AOF_FSYNC_EVERTSEC只损失1s的数据是不准确的

AOF_FSYNC_ALWAYS :每执行一个命令保存一次

每次执行完一个命令都会执行一次wirte指令和save指令,但是save是Redis主进程执行的所以主进程会阻塞

三种保存模式的对比图

image

image

AOF文件读取与数据还原

AOF文件内容如下

*2
 $6
 SELECT
 $1
 0
 *3
 $3
 SET
 $3
 key
 $5
 value
 *8
 $5
 RPUSH
 $4
 list
 $1
 1
 $1
 2
 $1
 3
 $1
 4
 $1
 5
 $1

我们可以看到在AOF文件内容中有一个SELECT 0 指令,该指令是为AOF文件指定要还原的数据库。

AOF文件的数据还原步骤如下:

① 开启一个伪客户端(fake cilent)

② 读取AOF的文件内容,并将其处理为命令,命令参数,参数个数该形式

③ 使伪客户端执行这些命令,直至所有命令执行完毕

这三步结束后,便会将AOF文件中的内容全部还原成数据库数据。在加载和还原期间只有订阅和发布功能能够使用,其他的都不能使用。

def READ_AND_LOAD_AOF():
 	# 打开并读取 AOF 文件
    file = open(aof_file_name)
 	while file.is_not_reach_eof():
         # 读入一条协议文本格式的 Redis 命令
        cmd_in_text = file.read_next_command_in_protocol_format()
         # 根据文本命令,查找命令函数,并创建参数和参数个数等对象
        cmd, argv, argc = text_to_command(cmd_in_text)
 		# 执行命令
		execRedisCommand(cmd, argv, argc)
 	# 关闭文件
	file.close()

AOF重写

对于一个Redis服务器来说,可能会接收几十万上千万的指令请求,如果此时将这些请求全部存入AOF文件,将会导致AOF文件不断庞大,对Redis和系统造成影响,于是为了将AOF文件进行压缩,便设计了AOF重写方法:AOF文件并不一定要写入所有的客户端指令只要保证前后状态一致即可,创建一个新的AOF文件替代原本的AOF文件,新AOF文件和原有的AOF文件对于数据库状态完全一样

实现原理:

对于下列命令集合我们可以发现,我们一开始创建了一个list[1,2,3,4],然后经过三次操作将其变为了list[1,2,3],那其实这四段命令我们可以直接压缩成一行也就是RPUSH list 1 2 3,这样即使的结果和前面的四次操作完全一致。

RPUSH list 1 2 3 4	 // [1, 2, 3, 4]
RPOP list			// [1, 2, 3]
LPOP list			// [2, 3]
LPUSH list 1		// [1, 2, 3]
def AOF_REWRITE(tmp_tile_name):
 	f = create(tmp_tile_name)
 	# 遍历所有数据库
    for db in redisServer.db:
         # 如果数据库为空,那么跳过这个数据库
        if db.is_empty(): continue
         # 写入 SELECT 命令,用于切换数据库
        f.write_command("SELECT " + db.number)
     	# 遍历所有键
        for key in db:
     		# 如果键带有过期时间,并且已经过期,那么跳过这个键
            if key.have_expire_time() and key.is_expired(): continue
    	    if key.type == String:
         		# 用 SET key value 命令来保存字符串键
    			value = get_value_from_string(key)
     			f.write_command("SET " + key + value)
     		elif key.type == List:
                 # 用 RPUSH key item1 item2 ... itemN 命令来保存列表键
                 item1, item2, ..., itemN = get_item_from_list(key)
     			f.write_command("RPUSH " + key + item1 + item2 + ... + itemN)
     		elif key.type == Set:
                 # 用 SADD key member1 member2 ... memberN 命令来保存集合键
                member1, member2, ..., memberN = get_member_from_set(key)
                f.write_command("SADD " + key + member1 + member2 + ... + memberN)
     		elif key.type == Hash:
                 # 用 HMSET key field1 value1 field2 value2 ... fieldN valueN 命令来保存哈希键
                field1, value1, field2, value2, ..., fieldN, valueN =\
                get_field_and_value_from_hash(key)
                f.write_command("HMSET " + key + field1 + value1 + field2 + value2 +\
                ... + fieldN + valueN)
     		elif key.type == SortedSet:
                 # 用 ZADD key score1 member1 score2 member2 ... scoreN memberN
                 # 命令来保存有序集键
                 score1, member1, score2, member2, ..., scoreN, memberN = \
                 get_score_and_member_from_sorted_set(key)
                 f.write_command("ZADD " + key + score1 + member1 + score2 + member2 +\
                 ... + scoreN + memberN)
             else:
                 raise_type_error()
                 # 如果键带有过期时间,那么用 EXPIREAT key time 命令来保存键的过期时间
    		if key.have_expire_time():
     			f.write_command("EXPIREAT " + key + key.expire_time_in_unix_timestamp())
         # 关闭文件
        f.close()

从上面的代码我们可以总结出以下步骤:

  • 遍历数据库,如果数据库为空则跳过,否则进入进行key遍历
  • 对所有的key进行遍历,如果key过期了则跳过,否则直接根据类型获取key的值,然后通过set的方式将其写入AOF文件中
  • 如果key有过期时间则给其赋予过期时间
  • 关闭文件写入
后台AOF重写

通过上面的AOF重写我们可以得知AOF重写是阻塞的,Redis也为AOF重写fork了一个子进程进行重写的处理。

image

① 执行BGREWRITEAOF指令,父进程fork出一个子进程来进行AOF重写操作

② 同时创建aof_rewrite_buf来缓存在重写过程中,执行的新的命令,父进程会将执行的命令同时放入aof_rewrite_buf和aof_buf中,保证不管是重写失败还是重写过程中都不会发生丢失数据的情况。

③ 子进程根据aof_rewrite_buf将重写后的指令写入新的AOF文件中

④ 当前重写全部执行完成后向父进程发送一个通知

⑤ 父进程将新的AOF文件与旧的AOF文件替换,完成重写

重写的自动触发条件:

重写可以通过手动命令bgrewriteaof进行,也可以自动进行不过要符合以下条件

  • 没有BGSAVE在执行
  • 没有SAVE在执行
  • 没有BGREWRITEAOF在执行
  • 当前AOF文件大小大于aof_rewrite_min_size(重写触发最小值 默认1mb)
  • 比较当前AOF文件和最后一次AOF文件重写的大小之间的比例是否超过一倍(比如当前AOF文件是2MB,重写时的文件是1MB,此时就超过一倍了)

符合以上条件则会进行自动的AOF重写。

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

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

相关文章

Java八股-3

面向对象基础 面向对象与面向过程的区别 面向过程把解决问题的过程拆解成一个个方法,通过一个个方法的执行来解决问题。 面向对象会先抽象出对象,再用对象执行方法的方式来解决问题。 面向对象开发的程序一般更易维护、易复用、易扩展。 创建一个对…

linux内核驱动-在内核代码里添加设备结点

linux中,一切皆文件 我们在用户层用一些系统函数(如:fopen等等)时,会进入内核,内核会在字符注册了的设备号链表中查找。如果找到就运行我们写的设备文件的(驱动)函数 我们在前面已经…

1.2.4 采用Java配置类管理Bean

本实战将演示如何使用Java配置类管理Bean,实现基于注解的IoC容器的配置。 创建新包 在net.huawei.spring根包里创建day04子包。 创建杀龙任务类 在day04子包里创建SlayDragonQuest类。在该类上不添加Component注解。 创建勇敢骑士类 在day04子包里创建BraveKnight…

07 Php学习:运算符

PHP 算术运算符 在 PHP 中,算术运算符用于执行基本的数学运算,包括加法、减法、乘法、除法、取余数,负数运算、取反和并置运算。以下是这些运算符的详细解释和示例: 加法运算符 :用于将两个数值相加。 $a 5; $b 3;…

【复现】用友NC-Cloud文件上传漏洞_70

目录 一.概述 二 .漏洞影响 三.漏洞复现 1. 漏洞一: 四.修复建议: 五. 搜索语法: 六.免责声明 一.概述 用友NC Cloud大型企业数字化平台,深度应用新一代数字智能技术,完全基于云原生架构,打造开放、…

【Qt 学习笔记】Qt信号和槽的其他说明及Lambda表达式

博客主页:Duck Bro 博客主页系列专栏:Qt 专栏关注博主,后期持续更新系列文章如果有错误感谢请大家批评指出,及时修改感谢大家点赞👍收藏⭐评论✍ Qt信号和槽的其他说明及Lambda表达式 文章编号:Qt 学习笔记…

团结引擎+OpenHarmony 2 xlua编译篇

文章目录 前言一、下载 xlua 源码二、OpenHarmony SDK三、开干 前言 提示:我们的 app 鸿蒙化过程 需要用到 xlua ,目前没有适配 OpenHarmony 平台,所以需要重新编译一下。编译有多种方式,但是我只会这一种 就是使用 cmake。 一、下载 xlua 源…

【UE5 C++】访问修饰符public/protected/private继承

限制类与类之间访问级别的方法 public 在类中创建一个public的部分,即" public: ",public之下的所有内容都是公共的,可以在类之间访问 private(最常见) 如若没有指明修饰符,则默认为private 不能在类的外…

如何保证消息不丢失?——使用rabbitmq的死信队列!

如何保证消息不丢失?——使用rabbitmq的死信队列! 1、什么是死信 在 RabbitMQ 中充当主角的就是消息,在不同场景下,消息会有不同地表现。 死信就是消息在特定场景下的一种表现形式,这些场景包括: 消息被拒绝访问&am…

what is tty?

waht is tty? 黑话:TTY 为什么使用Linux的时候CtrlC就会终止一个命令运行,ta是如何设置的? stty -a 桌面切换 CTRL ALT F1 – 锁屏 CTRL ALT F2 – 桌面环境 CTRL ALT F3 – TTY3 CTRL ALT F4 – TTY4 CTRL ALT F5 – TTY5 CTRL ALT F6 – TTY6

《C++程序设计》阅读笔记【7-堆和拷贝构造函数】

🌈个人主页:godspeed_lucip 🔥 系列专栏:《C程序设计》阅读笔记 本文对应的PDF源文件请关注微信公众号程序员刘同学,回复C程序设计获取下载链接。 1 堆与拷贝构造函数1.1 概述1.2 分配堆对象1.3 拷贝构造函数1.3.1 默…

python画图Matplotlib和Seaborn

python画图Matplotlib和Season 一、Matplotlib1、介绍2、安装3、内容二、Seaborn1、介绍2、安装3、内容一、Matplotlib Matplotlib官网 1、介绍 Matplotlib 是一个 Python 的绘图库,用于创建高质量的二维图表和一些基本的三维图表。它广泛应用于科学计算、数据分析、工程学和…

面试经典算法系列之双指针1 -- 合并两个有序数组

面试经典算法题1 – 合并两个有序数组 LeetCode.88 公众号:阿Q技术站 问题描述 给你两个按 非递减顺序 排列的整数数组 nums1 和 nums2,另有两个整数 m 和 n ,分别表示 nums1 和 nums2 中的元素数目。 请你 合并 nums2 到 nums1 中&#…

LangChain入门:17.使用 ConversationChain实现对话记忆功能

在默认情况下,无论是 LLM 还是代理都是无状态的,每次模型的调用都是独立于其他交互的。也就是说,我们每次通过 API 开始和大语言模型展开一次新的对话,它都不知道你其实昨天或者前天曾经和它聊过天了。 你肯定会说,不可…

全新智慧公厕解决方案,一键查看附近公厕情况

随着城市化进程的不断加快,人口密集地区的公共厕所需求日益增长,而传统的公厕管理方式已经无法满足人们对卫生、便利的需求。为了提升公共卫生设施的管理水平和服务质量,一家智能科技公司近日推出了全新智慧公厕解决方案,通过手机…

Cohere推出全新升级版RAG大型AI模型:支持中文,搭载1040亿参数,现开源其权重!

4月5日,知名类ChatGPT平台Cohere在其官方网站上发布了一款全新的模型——Command R。 据官方消息,Command R拥有1040亿个参数,并且支持包括英语、中文、法语、德语在内的10种语言。这一模型的显著特点之一在于其对内置的RAG(检索增…

【日期】获取当天以及未来三天的日期和周几

// 获取当天以及未来三天的日期和周几getDates() {const today new Date();const dayOfWeek ["星期日", "星期一", "星期二", "星期三", "星期四", "星期五", "星期六"];const todayDate today.toDa…

FreeRTOS任务切换学习

FreeRTOS任务切换学习 所谓任务切换,就是CPU寄存器的切换。假设当由任务A切换到任务B时,主要分为两步: 1:需暂停任务A的执行,并将此时任务A的寄存器保存到任务堆栈,这个过程叫做保存现场; 2&am…

Git 安装和配置

下载 Git 网址: https://git-scm.com/download 安装 Git 双击安装包, 开始安装. 修改安装路径, 选择非中文无空格路径: 开始安装: 安装成功: 配置 Git 安装完成后, 在任意文件夹内, 右键, 可以显示两个 Git 选项, 就说明安装成功了.

浅聊java集合框架中的java.util.LinkedList

java集合框架总览 Java集合框架是一个用来代表和操纵集合的统一架构,它为管理和组织对象的集合提供了一组类和接口。这个框架包含三个主要部分:接口、实现和算法。 接口: Collection:这是集合框架的根接口,定义了集…