当前位置: 首页 > news >正文

Redis事务机制详解与Springboot项目中的使用

Redis 的事务机制允许将多个命令打包在一起,作为一个原子操作来执行。虽然 Redis 的事务与关系型数据库的事务有所不同,但它仍然提供了一种确保多个命令顺序执行的方式。以下是 Redis 事务机制的详细解析:


1. Redis 事务的基本概念

Redis 事务通过以下四个命令实现:

  • MULTI:开启一个事务。
  • EXEC:执行事务中的所有命令。
  • DISCARD:取消事务,放弃所有已入队的命令。
  • WATCH:监视一个或多个键,如果在事务执行前这些键被修改,则事务不会执行。

Redis 事务的核心思想是将多个命令放入一个队列中,然后一次性、按顺序执行这些命令。


2. Redis 事务的工作流程

2.1 开启事务

使用 MULTI 命令开启一个事务。开启事务后,所有后续的命令都会被放入一个队列中,而不是立即执行。

127.0.0.1:6379> MULTI
OK
2.2 命令入队

在事务开启后,所有命令都会被放入队列中,等待执行。例如:

127.0.0.1:6379> SET key1 value1
QUEUED
127.0.0.1:6379> SET key2 value2
QUEUED
2.3 执行事务

使用 EXEC 命令执行事务中的所有命令。Redis 会按顺序执行队列中的命令,并返回每个命令的执行结果。

127.0.0.1:6379> EXEC
1) OK
2) OK
2.4 取消事务

如果在事务执行前需要取消事务,可以使用 DISCARD 命令。这会清空事务队列并退出事务。

127.0.0.1:6379> DISCARD
OK

3. Redis 事务的特性

3.1 原子性

Redis 事务是原子的,这意味着事务中的所有命令要么全部执行,要么全部不执行。但是,Redis 事务不支持回滚(rollback)。如果在事务执行过程中某个命令失败,后续命令仍然会继续执行。

3.2 隔离性

Redis 事务是隔离的,事务中的命令在 EXEC 执行之前不会被其他客户端看到。其他客户端只有在事务提交后(即 EXEC 执行后)才能看到事务的结果。

3.3 无回滚机制

Redis 事务不支持回滚。如果在事务执行过程中某个命令失败(例如语法错误),Redis 不会自动回滚已经执行的命令。这与关系型数据库的事务机制不同。

3.4 命令入队

在事务开启后,所有命令都会被放入队列中,而不是立即执行。只有在 EXEC 命令被调用时,队列中的命令才会被执行。


4. WATCH 命令

WATCH 命令用于监视一个或多个键。如果在事务执行前这些键被其他客户端修改,则事务不会执行。WATCH 提供了一种乐观锁机制,用于解决并发问题。

4.1 使用 WATCH
127.0.0.1:6379> WATCH key1
OK
127.0.0.1:6379> MULTI
OK
127.0.0.1:6379> SET key1 value1
QUEUED
127.0.0.1:6379> EXEC
(nil)  # 如果 key1 被其他客户端修改,事务不会执行
4.2 取消 WATCH

使用 UNWATCH 命令可以取消对所有键的监视。

127.0.0.1:6379> UNWATCH
OK

5. Redis 事务的局限性

5.1 不支持回滚

Redis 事务不支持回滚。如果在事务执行过程中某个命令失败,Redis 不会自动回滚已经执行的命令。

5.2 命令错误与运行时错误
  • 命令错误:如果事务中的某个命令存在语法错误(例如命令不存在),则整个事务都不会执行。
  • 运行时错误:如果事务中的某个命令在执行时出错(例如对字符串执行 INCR 操作),则只有该命令会失败,其他命令仍然会执行。
5.3 性能问题

Redis 事务会将所有命令放入队列中,直到 EXEC 执行时才一次性执行。如果事务中包含大量命令,可能会导致内存占用过高。


6. Redis 事务的应用场景

6.1 批量操作

当需要一次性执行多个命令时,可以使用事务来确保这些命令按顺序执行。

6.2 乐观锁

通过 WATCH 命令可以实现乐观锁机制,确保在事务执行前监视的键没有被修改。

6.3 原子性操作

虽然 Redis 事务不支持回滚,但它仍然可以确保多个命令的原子性执行。


7. Redis 事务与 Lua 脚本的对比

Redis 事务和 Lua 脚本都可以用于实现原子性操作,但两者有以下区别:

  • 事务:适合简单的批量操作,但不支持复杂的逻辑。
  • Lua 脚本:适合复杂的业务逻辑,支持条件判断、循环等操作,且脚本在服务器端原子执行。

8. Redis 事务的示例

以下是一个完整的 Redis 事务示例:

# 监视 key1
127.0.0.1:6379> WATCH key1
OK# 开启事务
127.0.0.1:6379> MULTI
OK# 命令入队
127.0.0.1:6379> SET key1 value1
QUEUED
127.0.0.1:6379> SET key2 value2
QUEUED# 提交事务
127.0.0.1:6379> EXEC
1) OK
2) OK

在 Spring Boot 中使用 Redis 事务机制时,可以通过 RedisTemplateStringRedisTemplate 来操作 Redis 事务。Spring Data Redis 提供了对 Redis 事务的支持,允许你在 Spring 应用中方便地使用 Redis 事务。

9. 在 Spring Boot 中使用 Redis 事务

9.1 配置 RedisTemplate

首先,确保在 Spring Boot 项目中配置了 RedisTemplateStringRedisTemplate

@Configuration
public class RedisConfig {@Beanpublic RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory redisConnectionFactory) {RedisTemplate<String, Object> template = new RedisTemplate<>();template.setConnectionFactory(redisConnectionFactory);template.setKeySerializer(new StringRedisSerializer());template.setValueSerializer(new GenericJackson2JsonRedisSerializer());return template;}
}
9.2 使用 Redis 事务

在 Spring Boot 中,可以通过 RedisTemplateexecute 方法来执行事务操作。execute 方法接受一个 SessionCallbackRedisCallback 参数,用于在事务中执行多个命令。

@Service
public class RedisTransactionService {@Autowiredprivate RedisTemplate<String, Object> redisTemplate;public void executeTransaction() {redisTemplate.execute(new SessionCallback<Object>() {@Overridepublic Object execute(RedisOperations operations) throws DataAccessException {// 开启事务operations.multi();// 执行多个命令operations.opsForValue().set("key1", "value1");operations.opsForValue().set("key2", "value2");// 提交事务return operations.exec();}});}
}
9.3 使用 WATCH 命令

WATCH 命令用于监视一个或多个键,如果在事务执行前这些键被修改,则事务不会执行。可以通过 RedisTemplatewatch 方法来实现。

@Service
public class RedisTransactionService {@Autowiredprivate RedisTemplate<String, Object> redisTemplate;public void executeTransactionWithWatch() {redisTemplate.execute(new SessionCallback<Object>() {@Overridepublic Object execute(RedisOperations operations) throws DataAccessException {// 监视 key1operations.watch("key1");// 开启事务operations.multi();// 执行多个命令operations.opsForValue().set("key1", "value1");operations.opsForValue().set("key2", "value2");// 提交事务return operations.exec();}});}
}
9.4. 事务的异常处理

在 Redis 事务中,如果某个命令执行失败,事务不会回滚,而是继续执行后续命令。因此,需要在代码中处理可能的异常情况。

@Service
public class RedisTransactionService {@Autowiredprivate RedisTemplate<String, Object> redisTemplate;public void executeTransactionWithExceptionHandling() {redisTemplate.execute(new SessionCallback<Object>() {@Overridepublic Object execute(RedisOperations operations) throws DataAccessException {try {// 开启事务operations.multi();// 执行多个命令operations.opsForValue().set("key1", "value1");operations.opsForValue().set("key2", "value2");// 提交事务return operations.exec();} catch (Exception e) {// 处理异常operations.discard();throw e;}}});}
}
9.5. 使用注解驱动的事务管理

Spring Data Redis 支持通过 @Transactional 注解来管理 Redis 事务。需要在配置类中启用事务管理。

@Configuration
@EnableTransactionManagement
public class RedisConfig {@Beanpublic RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory redisConnectionFactory) {RedisTemplate<String, Object> template = new RedisTemplate<>();template.setConnectionFactory(redisConnectionFactory);template.setKeySerializer(new StringRedisSerializer());template.setValueSerializer(new GenericJackson2JsonRedisSerializer());template.setEnableTransactionSupport(true); // 启用事务支持return template;}@Beanpublic PlatformTransactionManager transactionManager(RedisConnectionFactory redisConnectionFactory) {return new DataSourceTransactionManager();}
}

然后在 Service 类中使用 @Transactional 注解来标记事务方法。

@Service
public class RedisTransactionService {@Autowiredprivate RedisTemplate<String, Object> redisTemplate;@Transactionalpublic void executeTransactionWithAnnotation() {redisTemplate.opsForValue().set("key1", "value1");redisTemplate.opsForValue().set("key2", "value2");}
}

总结

在 Spring Boot 中使用 Redis 事务机制时,可以通过 RedisTemplateexecute 方法手动管理事务,也可以通过 @Transactional 注解实现声明式事务管理。使用 WATCH 命令可以确保事务的原子性,避免竞态条件。在实际应用中,需要根据业务需求选择合适的事务管理方式,并注意异常处理和性能优化。

相关文章:

Redis事务机制详解与Springboot项目中的使用

Redis 的事务机制允许将多个命令打包在一起&#xff0c;作为一个原子操作来执行。虽然 Redis 的事务与关系型数据库的事务有所不同&#xff0c;但它仍然提供了一种确保多个命令顺序执行的方式。以下是 Redis 事务机制的详细解析&#xff1a; 1. Redis 事务的基本概念 Redis 事…...

DeepSeek-R1,用Ollama跑起来

# DeepSeek-R1横空出世&#xff0c;超越OpenAI-o1&#xff0c;教你用Ollama跑起来 使用Ollama在本地运行DeepSeek-R1的操作指南。 DeepSeek-R1作为第一代推理模型&#xff0c;在数学、代码和推理任务上表现优异&#xff0c;与OpenAI-o1模型不相上下。 将此类模型部署到本地&am…...

Leecode刷题C语言之组合总和②

执行结果:通过 执行用时和内存消耗如下&#xff1a; int** ans; int* ansColumnSizes; int ansSize;int* sequence; int sequenceSize;int** freq; int freqSize;void dfs(int pos, int rest) {if (rest 0) {int* tmp malloc(sizeof(int) * sequenceSize);memcpy(tmp, seque…...

YOLOv8改进,YOLOv8检测头融合DynamicHead,并添加小目标检测层(四头检测),适合目标检测、分割等,全网独发

摘要 作者提出一种新的检测头,称为“动态头”,旨在将尺度感知、空间感知和任务感知统一在一起。如果我们将骨干网络的输出(即检测头的输入)视为一个三维张量,其维度为级别 空间 通道,这样的统一检测头可以看作是一个注意力学习问题,直观的解决方案是对该张量进行全自…...

【PyQt】QThread快速创建多线程任务

pyqt通过QThread快速创建多线程任务 在 PyQt5 中使用多线程时&#xff0c;需要注意 GUI 线程&#xff08;主线程&#xff09; 和 工作线程 的分离。PyQt5 的主线程负责处理 GUI 事件&#xff0c;如果在主线程中执行耗时任务&#xff0c;会导致界面卡顿甚至无响应。因此&#x…...

智能码二维码的成本效益分析

以下是智能码二维码的成本效益分析&#xff1a; 成本方面 硬件成本 标签成本&#xff1a;二维码标签本身价格低廉&#xff0c;即使进行大规模应用&#xff0c;成本也相对较低。如在智能仓储中&#xff0c;塑料托盘加二维码方案的标签成本几乎可以忽略不计4。扫描设备成本&…...

企业财务管理系统的需求设计和实现

该作者的原创文章目录&#xff1a; 生产制造执行MES系统的需求设计和实现 企业后勤管理系统的需求设计和实现 行政办公管理系统的需求设计和实现 人力资源管理HR系统的需求设计和实现 企业财务管理系统的需求设计和实现 董事会办公管理系统的需求设计和实现 公司组织架构…...

Springboot集成Swagger和Springdoc详解

Springboot2.x集成Swagger21. Springboot匹配版本2.7.0~2.7.18(其它版本需要自己去调试匹配)2. 首先导入Swagger2匹配的依赖项3. 导入依赖后创建配置文件SwaggerConfig4. Swagger集成完后,接下来接口的配置Springboot3.x集成Springdoc1. Springboot3.x依赖Springdoc配置2. 在…...

类和对象(4)——多态:方法重写与动态绑定、向上转型和向下转型、多态的实现条件

目录 1. 向上转型和向下转型 1.1 向上转型 1.2 向下转型 1.3 instanceof关键字 2. 重写&#xff08;overidde&#xff09; 2.1 方法重写的规则 2.1.1 基础规则 2.1.2 深层规则 2.2 三种不能重写的方法 final修饰 private修饰 static修饰 3. 动态绑定 3.1 动态绑…...

ui-automator定位官网文档下载及使用

一、ui-automator定位官网文档简介及下载 AndroidUiAutomator&#xff1a;移动端特有的定位方式&#xff0c;uiautomator是java实现的&#xff0c;定位类型必须写成java类型 官方地址&#xff1a;https://developer.android.com/training/testing/ui-automator.html#ui-autom…...

董事会办公管理系统的需求设计和实现

该作者的原创文章目录&#xff1a; 生产制造执行MES系统的需求设计和实现 企业后勤管理系统的需求设计和实现 行政办公管理系统的需求设计和实现 人力资源管理HR系统的需求设计和实现 企业财务管理系统的需求设计和实现 董事会办公管理系统的需求设计和实现 公司组织架构…...

ESP32和STM32在处理中断方面的区别

为了通俗地讲解ESP32和STM32在处理中断方面的区别&#xff0c;我们可以把它们想象成两个不同的“智能管家”系统&#xff0c;各自负责管理一个家庭&#xff08;即嵌入式项目&#xff09;的各种任务。我们将重点放在如何处理突发事件&#xff08;即中断&#xff09;上。 ESP32 …...

零售业革命:改变行业的顶级物联网用例

mpro5 产品负责人Ruby Whipp表示&#xff0c;技术进步持续重塑零售业&#xff0c;其中物联网&#xff08;IoT&#xff09;正引领这一变革潮流。 研究表明&#xff0c;零售商们正在采用物联网解决方案&#xff0c;以提升运营效率并改善顾客体验。这些技术能够监控运营的各个方面…...

字符串算法笔记

字符串笔记 说到字符串,首先我们要注意的就是字符串的输入以及输出,因为字符串的输入格式以及要求也分为很多种,我们就来说几个比较常见的格式 g e t s gets gets 我们先来说这个函数的含义...

在Ubuntu上用Llama Factory命令行微调Qwen2.5的简单过程

半年多之前写过一个教程&#xff1a;在Windows上用Llama Factory微调Llama 3的基本操作_llama-factory windows-CSDN博客 如果用命令行做的话&#xff0c;前面的步骤可以参考上面这个博客。安装好环境后&#xff0c; 用自我认知数据集微调Lora模块&#xff1a;data/identity.j…...

ThinkPhp伪静态设置后,访问静态资源也提示找不到Controller

ThinkPhp没有配置伪静态时&#xff0c;除了默认的IndexController能访问&#xff0c;其他路由Controller都访问不到&#xff0c;提示404错误。配置了伪静态后就解决了这个问题。 但是当我的ThinkPhp后台项目中有静态资源放在public目录&#xff08;或子目录&#xff09;中需要…...

JavaScript赋能智能网页设计

构建AI驱动的实时风格迁移系统 案例概述 本案例将实现一个基于深度学习的实时图像风格迁移系统&#xff0c;通过浏览器端神经网络推理实现以下高级特性&#xff1a; WebAssembly加速的ONNX模型推理 WebGL Shader实现的风格混合算法 WebRTC实时视频流处理 基于Web Workers的…...

基于STM32的阿里云智能农业大棚

目录 前言&#xff1a; 项目效果演示&#xff1a; 一、简介 二、硬件需求准备 三、硬件框图 四、CubeMX配置 4.1、按键、蜂鸣器GPIO口配置 4.2、ADC输入配置 4.3、IIC——驱动OLED 4.4、DHT11温湿度读取 4.5、PWM配置——光照灯、水泵、风扇 4.6、串口——esp8266模…...

80,【4】BUUCTF WEB [SUCTF 2018]MultiSQL

53&#xff0c;【3】BUUCTF WEB october 2019 Twice SQLinjection-CSDN博客 上面这个链接是我第一次接触二次注入 这道题也涉及了 对二次注入不熟悉的可以看看 BUUCTF出了点问题&#xff0c;打不开&#xff0c;以下面这两篇wp作为学习对象 [SUCTF 2018]MultiSQL-CSDN博客 …...

深入探索imi框架:PHP Swoole的高性能协程应用实践

摘要 本文将介绍 imi 框架&#xff0c;这是一个基于 PHP Swoole 的高性能协程应用开发框架。imi 支持 HttpApi、WebSocket、TCP 和 UDP 等多种服务类型&#xff0c;利用 Swoole 的优化技术&#xff0c;使得在处理请求时响应速度远超传统的 php-fpm 方式。通过丰富的代码示例&a…...

应用升级/灾备测试时使用guarantee 闪回点迅速回退

1.场景 应用要升级,当升级失败时,数据库回退到升级前. 要测试系统,测试完成后,数据库要回退到测试前。 相对于RMAN恢复需要很长时间&#xff0c; 数据库闪回只需要几分钟。 2.技术实现 数据库设置 2个db_recovery参数 创建guarantee闪回点&#xff0c;不需要开启数据库闪回。…...

ES6从入门到精通:前言

ES6简介 ES6&#xff08;ECMAScript 2015&#xff09;是JavaScript语言的重大更新&#xff0c;引入了许多新特性&#xff0c;包括语法糖、新数据类型、模块化支持等&#xff0c;显著提升了开发效率和代码可维护性。 核心知识点概览 变量声明 let 和 const 取代 var&#xf…...

CTF show Web 红包题第六弹

提示 1.不是SQL注入 2.需要找关键源码 思路 进入页面发现是一个登录框&#xff0c;很难让人不联想到SQL注入&#xff0c;但提示都说了不是SQL注入&#xff0c;所以就不往这方面想了 ​ 先查看一下网页源码&#xff0c;发现一段JavaScript代码&#xff0c;有一个关键类ctfs…...

SCAU期末笔记 - 数据分析与数据挖掘题库解析

这门怎么题库答案不全啊日 来简单学一下子来 一、选择题&#xff08;可多选&#xff09; 将原始数据进行集成、变换、维度规约、数值规约是在以下哪个步骤的任务?(C) A. 频繁模式挖掘 B.分类和预测 C.数据预处理 D.数据流挖掘 A. 频繁模式挖掘&#xff1a;专注于发现数据中…...

【网络安全产品大调研系列】2. 体验漏洞扫描

前言 2023 年漏洞扫描服务市场规模预计为 3.06&#xff08;十亿美元&#xff09;。漏洞扫描服务市场行业预计将从 2024 年的 3.48&#xff08;十亿美元&#xff09;增长到 2032 年的 9.54&#xff08;十亿美元&#xff09;。预测期内漏洞扫描服务市场 CAGR&#xff08;增长率&…...

为什么需要建设工程项目管理?工程项目管理有哪些亮点功能?

在建筑行业&#xff0c;项目管理的重要性不言而喻。随着工程规模的扩大、技术复杂度的提升&#xff0c;传统的管理模式已经难以满足现代工程的需求。过去&#xff0c;许多企业依赖手工记录、口头沟通和分散的信息管理&#xff0c;导致效率低下、成本失控、风险频发。例如&#…...

Java 加密常用的各种算法及其选择

在数字化时代&#xff0c;数据安全至关重要&#xff0c;Java 作为广泛应用的编程语言&#xff0c;提供了丰富的加密算法来保障数据的保密性、完整性和真实性。了解这些常用加密算法及其适用场景&#xff0c;有助于开发者在不同的业务需求中做出正确的选择。​ 一、对称加密算法…...

HBuilderX安装(uni-app和小程序开发)

下载HBuilderX 访问官方网站&#xff1a;https://www.dcloud.io/hbuilderx.html 根据您的操作系统选择合适版本&#xff1a; Windows版&#xff08;推荐下载标准版&#xff09; Windows系统安装步骤 运行安装程序&#xff1a; 双击下载的.exe安装文件 如果出现安全提示&…...

微信小程序云开发平台MySQL的连接方式

注&#xff1a;微信小程序云开发平台指的是腾讯云开发 先给结论&#xff1a;微信小程序云开发平台的MySQL&#xff0c;无法通过获取数据库连接信息的方式进行连接&#xff0c;连接只能通过云开发的SDK连接&#xff0c;具体要参考官方文档&#xff1a; 为什么&#xff1f; 因为…...

OPENCV形态学基础之二腐蚀

一.腐蚀的原理 (图1) 数学表达式&#xff1a;dst(x,y) erode(src(x,y)) min(x,y)src(xx,yy) 腐蚀也是图像形态学的基本功能之一&#xff0c;腐蚀跟膨胀属于反向操作&#xff0c;膨胀是把图像图像变大&#xff0c;而腐蚀就是把图像变小。腐蚀后的图像变小变暗淡。 腐蚀…...