百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术教程 > 正文

一文详解SpringBoot如何整合Redis事务操作?

mhr18 2024-11-18 14:41 22 浏览 0 评论

在SpringBoot整合Redis实战操作中,除了一些缓存功能以及临时存储功能之外,它还可以支持事务、流水线、订阅发布以及通过Lua脚本实现一些限流的功能。

这些功能在高并发场景中,最需要解决的问题就是数据一致性的问题,这个时候就可以用到Redis事务以及使用Lua脚本的方式来达到数据一致性的目的。下面我们就来看看,如何在Spring Boot项目中使用Redis的事务功能。

Redis事务操作

首先对于Redis的事务操作,是由一套组合的命令来完成的 watch……mulit……exec。也就是说要想完成一个Redis的事务操作那么就要在一个Redis连接中执行多个命令才能完成一套事务操作。如下图所示。

首先通过watch命令来监控Redis中的键值的变化,mulit命令用来开启事务操作,当事务操作开启之后客户端并不会马上去执行命令,而是将命令放入到一个命令执行队列中,这里需要注意,当命令被放入队列中的时候,会返回一些数据,但是并不是命令执行了。命令还是在命令队列中,这个时候,如果我们调用了get命令去获取相应的数据的时候,其实是获取不到的。

当执行了exec命令之后,首先需要做的事情就是通过watch去监听所执行的键值对是否发生了变化,即使出现了ABA的问题,也会被认为是发生了变化。如果判断到发生了变化,那么Redis就会取消事务的执行,反之就会执行队列中的命令,由于队列的特点FIFO所以命令会按照顺序执行。这个时候,这些命令操作会完成全部的执行,并且不会被其他的客户端所打断。这样就可以保证Redis事务操作下的数据一致性问题了。

在Spring Boot的中如何执行Redis事务操作

在Spring Boot中执行Redis事务操作,需要通过RedisTemplate或者是其他的Redis客户端工具类来实现。

引入依赖以及配置

<dependency>  
    <groupId>org.springframework.boot</groupId>  
    <artifactId>spring-boot-starter-data-redis</artifactId>  
</dependency>

在application.properties配置文件中添加Redis相关的配置

# application.properties  
spring.redis.host=localhost  
spring.redis.port=6379  
# 如果设置了密码  
spring.redis.password=yourpassword  
# 设置数据库索引,默认为0  
spring.redis.database=0  
# 连接池最大连接数(使用负值表示没有限制)  
spring.redis.jedis.pool.max-active=8  
# 连接池最大阻塞等待时间(使用负值表示没有限制)  
spring.redis.jedis.pool.max-wait=-1ms  
# 连接池中的最大空闲连接  
spring.redis.jedis.pool.max-idle=8  
# 连接池中的最小空闲连接  
spring.redis.jedis.pool.min-idle=0  
# 连接超时时间(毫秒)  
spring.redis.timeout=0

添加RedisConfig的配置文件用来完成序列化操作。

@Configuration  
public class RedisConfig {  

    @Bean  
    public RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory factory) {  
        RedisTemplate<String, Object> template = new RedisTemplate<>();  
        template.setConnectionFactory(factory);  

        // 使用Jackson2JsonRedisSerializer来序列化和反序列化redis的value值  
        Jackson2JsonRedisSerializer<Object> jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer<>(Object.class);  
        ObjectMapper om = new ObjectMapper();  
        om.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);  
        om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);  
        jackson2JsonRedisSerializer.setObjectMapper(om);  

        // String的序列化  
        StringRedisSerializer stringRedisSerializer = new StringRedisSerializer();  
				// 这些配置可以省略
        // Key采用String的序列化方式  
        template.setKeySerializer(stringRedisSerializer);  
        // value序列化方式采用jackson  
        template.setValueSerializer(jackson2JsonRedisSerializer);  
        // Hash的key也采用String的序列化方式  
        template.setHashKeySerializer(stringRedisSerializer);  
        // Hash的value序列化方式采用jackson  
        template.setHashValueSerializer(jackson2JsonRedisSerializer);  
        template.afterPropertiesSet();  

        return template;  
    }  
}

实现Redis服务

@Service  
public class RedisService {  

    @Autowired  
    private RedisTemplate<String, Object> redisTemplate;  

    public void executeRedisTransaction(final String key, final int decrementAmount) {  
        redisTemplate.execute(new SessionCallback<Object>() {  
            @Override  
            public Object execute(RedisOperations operations) throws DataAccessException {  
                operations.watch(key);  

                Integer currentStock = operations.opsForValue().get(key);  
                if (currentStock == null) {  
                    throw new RuntimeException("当前库存为null");  
                }  

                if (currentStock < decrementAmount) {  
                    throw new RuntimeException("库存不足");  
                }  

                TransactionOperations transaction = operations.multi();  
                transaction.opsForValue().decrement(key, decrementAmount);  

                return transaction.exec();  
            }  
        });  
    }  
}

上面这段代码中,就是实现了Redis的事务操作,首先watch方法检测key值的变化,第二步通过get方法获取到当前库存,然后判断当前库存是否为空,如果为空则返回库存为空,如果不为空则需要判断库存是否满足条件,如果不满足条件那么就抛出库存不足的异常,如果满足条件那么就使用mulit操作开启事务操作,并且,最后调用exec方法来完成事务的提交执行。

测试Redis事务

编写一个测试类,启动项目之后,可以模拟多线程调用该接口。就可以完成Redis事务执行的操作了。

@RestController
public class RedisController {

    @Autowired
    private RedisService redisService;

    @PostMapping("/deduct-stock")
    public ResponseEntity<String> deductStock(
            @RequestParam String productId, @RequestParam int amount) {
        try {
            redisService.executeRedisTransaction(productId, amount);
            return ResponseEntity.ok("库存扣减成功");
        } catch (RuntimeException e) {
            return ResponseEntity.error(e.getMessages());
        }
    }
}

总结

需要注意的是Redis事务操作并不像是关系型数据库那样具有原子性和隔离性,根据上面的介绍Redis事务操作其实是一组命令的集合,而这些命令是按它们在事务执行过程中的顺序来执行的,并且在事务执行过程中不会被其他的客户端打断。

但是,如果其中一个命令执行失败,Redis不会回滚之前已经执行成功的命令,这需要开发者自己处理。这是什么意思呢?就是说在事务中的命令要么全部成功要么全部失败,但是如果对于mulit、exec这样的命令如果执行失败了,那么依旧是无法保证事务操作的。

此外,由于Redis是单线程的,事务中的命令会串行执行,这保证了命令的顺序性。然而,如果某个命令执行时间非常长,那么它会阻塞后续命令的执行,这可能会影响到系统的性能和响应时间。

最后,Redis事务不支持回滚操作,因此开发者需要在编写事务逻辑时考虑如何处理错误和异常情况。

相关推荐

【推荐】一个开源免费、AI 驱动的智能数据管理系统,支持多数据库

如果您对源码&技术感兴趣,请点赞+收藏+转发+关注,大家的支持是我分享最大的动力!!!.前言在当今数据驱动的时代,高效、智能地管理数据已成为企业和个人不可或缺的能力。为了满足这一需求,我们推出了这款开...

Pure Storage推出统一数据管理云平台及新闪存阵列

PureStorage公司今日推出企业数据云(EnterpriseDataCloud),称其为组织在混合环境中存储、管理和使用数据方式的全面架构升级。该公司表示,EDC使组织能够在本地、云端和混...

对Java学习的10条建议(对java课程的建议)

不少Java的初学者一开始都是信心满满准备迎接挑战,但是经过一段时间的学习之后,多少都会碰到各种挫败,以下北风网就总结一些对于初学者非常有用的建议,希望能够给他们解决现实中的问题。Java编程的准备:...

SQLShift 重大更新:Oracle→PostgreSQL 存储过程转换功能上线!

官网:https://sqlshift.cn/6月,SQLShift迎来重大版本更新!作为国内首个支持Oracle->OceanBase存储过程智能转换的工具,SQLShift在过去一...

JDK21有没有什么稳定、简单又强势的特性?

佳未阿里云开发者2025年03月05日08:30浙江阿里妹导读这篇文章主要介绍了Java虚拟线程的发展及其在AJDK中的实现和优化。阅前声明:本文介绍的内容基于AJDK21.0.5[1]以及以上...

「松勤软件测试」网站总出现404 bug?总结8个原因,不信解决不了

在进行网站测试的时候,有没有碰到过网站崩溃,打不开,出现404错误等各种现象,如果你碰到了,那么恭喜你,你的网站出问题了,是什么原因导致网站出问题呢,根据松勤软件测试的总结如下:01数据库中的表空间不...

Java面试题及答案最全总结(2025版)

大家好,我是Java面试陪考员最近很多小伙伴在忙着找工作,给大家整理了一份非常全面的Java面试题及答案。涉及的内容非常全面,包含:Spring、MySQL、JVM、Redis、Linux、Sprin...

数据库日常运维工作内容(数据库日常运维 工作内容)

#数据库日常运维工作包括哪些内容?#数据库日常运维工作是一个涵盖多个层面的综合性任务,以下是详细的分类和内容说明:一、数据库运维核心工作监控与告警性能监控:实时监控CPU、内存、I/O、连接数、锁等待...

分布式之系统底层原理(上)(底层分布式技术)

作者:allanpan,腾讯IEG高级后台工程师导言分布式事务是分布式系统必不可少的组成部分,基本上只要实现一个分布式系统就逃不开对分布式事务的支持。本文从分布式事务这个概念切入,尝试对分布式事务...

oracle 死锁了怎么办?kill 进程 直接上干货

1、查看死锁是否存在selectusername,lockwait,status,machine,programfromv$sessionwheresidin(selectsession...

SpringBoot 各种分页查询方式详解(全网最全)

一、分页查询基础概念与原理1.1什么是分页查询分页查询是指将大量数据分割成多个小块(页)进行展示的技术,它是现代Web应用中必不可少的功能。想象一下你去图书馆找书,如果所有书都堆在一张桌子上,你很难...

《战场兄弟》全事件攻略 一般事件合同事件红装及隐藏职业攻略

《战场兄弟》全事件攻略,一般事件合同事件红装及隐藏职业攻略。《战场兄弟》事件奖励,事件条件。《战场兄弟》是OverhypeStudios制作发行的一款由xcom和桌游为灵感来源,以中世纪、低魔奇幻为...

LoadRunner(loadrunner录制不到脚本)

一、核心组件与工作流程LoadRunner性能测试工具-并发测试-正版软件下载-使用教程-价格-官方代理商的架构围绕三大核心组件构建,形成完整测试闭环:VirtualUserGenerator(...

Redis数据类型介绍(redis 数据类型)

介绍Redis支持五种数据类型:String(字符串),Hash(哈希),List(列表),Set(集合)及Zset(sortedset:有序集合)。1、字符串类型概述1.1、数据类型Redis支持...

RMAN备份监控及优化总结(rman备份原理)

今天主要介绍一下如何对RMAN备份监控及优化,这里就不讲rman备份的一些原理了,仅供参考。一、监控RMAN备份1、确定备份源与备份设备的最大速度从磁盘读的速度和磁带写的带度、备份的速度不可能超出这两...

取消回复欢迎 发表评论: