说一下redis事务底层原理

devtools/2025/3/10 19:39:48/

Redis事务

1. 事务的基本流程

Redis 事务通过 MULTIEXECWATCH 等命令实现,底层原理可以分为以下几个步骤:

(1) MULTI 命令
  • 当客户端发送 MULTI 命令时,Redis 会将客户端标记为“事务模式”。
  • 在事务模式下,客户端发送的所有命令不会立即执行,而是被放入一个队列(命令队列)中。
(2) 命令入队
  • MULTIEXEC 之间,客户端发送的所有命令都会被追加到事务队列中。
  • 这些命令不会立即执行,而是等待 EXEC 命令的触发。
(3) EXEC 命令
  • 当客户端发送 EXEC 命令时,Redis 会依次执行事务队列中的所有命令。
  • 执行过程中,所有命令是原子的,不会被其他客户端的命令打断。
(4) WATCH 命令
  • WATCH 命令用于实现乐观锁。
  • 当客户端对一个或多个键执行 WATCH 后,如果在 EXEC 执行之前,这些键被其他客户端修改,则当前事务会失败(返回 nil

watch我们可以指定监听一个键和多个键,然后exec批量执行

WATCH key [key ...]

2. 事务的原子性

  • Redis 事务的原子性是通过单线程模型实现的。
  • Redis 是单线程的,所有命令都是顺序执行的。在 EXEC 执行时,事务队列中的命令会连续执行,不会被其他客户端的命令打断。

3. 事务的一致性

  • Redis 事务的一致性是通过 WATCH 机制实现的。
  • 如果 WATCH 的键在事务执行期间被修改,事务会失败,从而保证数据的一致性。

4. 事务的局限性

  • 不支持回滚:如果事务中的某个命令失败,其他命令仍然会执行,Redis 不会自动回滚。
  • 部分失败问题:事务中的命令可能会部分成功、部分失败。
  • 性能开销WATCH 机制会增加额外的性能开销。

Redis 7 对事务的优化

Redis 7 在事务机制上并没有完全改变底层实现,但引入了一些优化和改进:

1. 性能优化

  • Redis 7 对事务的执行流程进行了优化,减少了事务模式下的性能开销。
  • 通过改进命令队列的处理方式,提高了事务的执行效率。

2. Lua 脚本的增强

  • Redis 7 对 Lua 脚本的支持进行了增强,使得 Lua 脚本可以更好地与事务结合使用。
  • Lua 脚本在 Redis 7 中的执行效率更高,同时支持更多的 Redis 命令。

3. 更好的错误处理

  • Redis 7 改进了事务中的错误处理机制,使得事务失败时的错误信息更加清晰。
  • 如果事务中的某个命令失败,Redis 7 会返回更详细的错误信息,方便排查问题。

4. 功能增强

  • Redis 7 引入了更多的命令和功能,可以与事务结合使用。
  • 例如,Redis 7 支持更多的数据类型和操作,使得事务可以处理更复杂的场景。

Redis 事务的底层实现细节

1. 命令队列

  • 在事务模式下,Redis 会为每个客户端维护一个命令队列。
  • 所有在 MULTIEXEC 之间发送的命令都会被追加到队列中。

2. 事务执行

  • EXEC 命令被触发时,Redis 会依次执行命令队列中的所有命令。
  • 执行过程中,Redis 会保证命令的原子性,不会被其他客户端的命令打断。

3. WATCH 机制

  • WATCH 命令会监视一个或多个键。
  • 如果在 EXEC 执行之前,这些键被其他客户端修改,则当前事务会失败。
  • WATCH 的实现基于 Redis 的键空间通知机制。

总结

  • Redis 事务的底层原理 是基于 MULTI/EXEC/WATCH 机制,通过命令队列和乐观锁实现原子性和一致性。
  • Redis 7 在事务机制上进行了性能优化和功能增强,但底层实现并没有本质变化。
  • Redis 事务的局限性 包括不支持回滚、部分失败问题和性能开销。
  • 如果需要更强大的事务支持,可以结合 Lua 脚本或使用支持 ACID 事务的数据库

Redis+Lua脚本实现手动回滚补偿

我们每一步执行失败,我们就依次撤销前面的操作

可惜这个并不是真正的acid,我们的mysql执行事务的时候宕机了,它的事务没有提交所以数据并不会进到mysql里面

redis是人为控制的,所以我们执行lua脚本的时候宕机了,我们之前事务中执行的操作数据仍然进去了,这个是我们无法解决的

local key1 = KEYS[1]
local key2 = KEYS[2]
local key3 = KEYS[3]
local value = ARGV[1]-- 记录原始值
local original_value1 = redis.call('GET', key1)
local original_value2 = redis.call('GET', key2)
local original_value3 = redis.call('GET', key3)-- 第一步操作
redis.call('SET', key1, value)-- 第二步操作
if redis.call('EXISTS', key2) == 0 then-- 手动回滚第一步操作redis.call('SET', key1, original_value1)return "Key2 does not exist"
end
redis.call('SET', key2, value)-- 第三步操作
if redis.call('EXISTS', key3) == 0 then-- 手动回滚前两步操作redis.call('SET', key1, original_value1)redis.call('SET', key2, original_value2)return "Key3 does not exist"
end
redis.call('SET', key3, value)return "Transaction successful"

http://www.ppmy.cn/devtools/166086.html

相关文章

Linux驱动学习之平台总线设备树驱动模型

0.步骤零:通用操作指针变量 static int major 0; //主设备号 static struct class *sr501_class; //定义类 static struct gpio_desc *sr501_gpio; //GPIO结构体操作指针 static int sr501_irq; …

Apache SeaTunnel 人物专访 | 张东浩:从使用者到Committer的开源历程

时光飞逝,转眼间,Apache SeaTunnel 社区已经成为顶级项目快两周年了,其社区贡献者和用户群体也日益壮大。SeaTunnel 凭借其高性能和插件灵活的特性,已经成为国内众多知名企业数据同步的基础工具。作为 SeaTunnel 的贡献者&#xf…

C++之序列容器(vector,list,dueqe)

1.大体对比 在软件开发的漫长历程中,数据结构与算法始终占据着核心地位,犹如大厦的基石,稳固支撑着整个程序的运行。在众多编程语言中,数据的存储与管理方式各有千秋,而 C 凭借其丰富且强大的工具集脱颖而出&#xff…

深入探索 Django 内置的 User 模型及其自定义扩展

深入探索 Django 内置的 User 模型及其自定义扩展 在 Django 框架中,内置的 User 模型是处理用户认证和授权的核心组件。它提供了一系列预定义的属性和方法,使得开发者能够轻松管理用户信息、进行用户认证以及控制用户权限。本文将详细介绍 Django 内置…

批量在 Word 的指定位置插入页,如插入封面、末尾插入页面

我们经常会碰到需要在 Word 文档中插入新的页面的需求,比如在 Word 文档末尾插入一个广告页、给 Word 文档插入一个说明封面,在 Word 文档的中间位置插入新的页面等等。相信这个操作对于大部分小伙伴来说都不难,难的是同时给多个 Word 文档插…

视频输入设备-V4L2的开发流程简述

一、摄像头的工作原理与应用 基本概念 V4L2的全称是Video For Linux Two,其实指的是V4L的升级版,是linux系统关于视频设备的内核驱动,同时V4L2也包含Linux系统下关于视频以及音频采集的接口,只需要配合对应的视频采集设备就可以实…

蓝桥杯嵌入式组第七届省赛题目解析+STM32G431RBT6实现源码

文章目录 1.题目解析1.1 分而治之,藕断丝连1.2 模块化思维导图1.3 模块解析1.3.1 KEY模块1.3.2 ADC模块1.3.3 IIC模块1.3.4 UART模块1.3.5 LCD模块1.3.6 LED模块1.3.7 TIM模块 2.源码3.第七届题目 前言:STM32G431RBT6实现嵌入式组第七届题目解析源码&…

阿里云操作系统(AliOS)

引言 阿里云操作系统(AliOS)是阿里巴巴集团专为物联网(IoT)和智能设备开发的操作系统,致力于为智能汽车、智能家居、工业设备等提供高效、安全、智能化的解决方案。作为一款云端一体的操作系统,AliOS深度融…