Java开发 - Redis事务怎么用?

前言

最近博主感觉捅了Redis窝,从Redis主从,哨兵,集群,集群原理纷纷讲了一遍,不知道大家都学会了多少,想着送佛送到西,不如再添一把火,所以今天带给大家的博客是Redis事务!有人要问,什么?Redis也有事务?没错,Redis也有类似于MySQL的事务,不要惊奇,这也不是刚刚有,只是很多人平时不常用罢了。

事务

事务我们知道,是存在于关系型数据库中的一种数据处理手段,主要责任是数据的CRUD,还包括了事务提交和回滚。那么在Redis里,有没有一种机制也能处理数据的提交和回滚呢?答案是肯定的,那就是Redis事务,它是一个类原子的隔离操作,将一系列指令按需排队并顺序执行,期间不会被其他指令插队。但,Redis事务在运行时无法预测,所以也无法回滚。

指令

在Redis事务中,共有三大指令:

  • multi:开启事务
  • exec:执行事务
  • discard:取消事务

在multi之后开启事务,此时可以set多个key,这些key将按照顺序存入先进先出的队列,exec执行事务,这些事务将按照入队的顺序执行,在这些事务执行过程中,不受其他的插入指令影响。在组队阶段使用discard指令,由于未使用exec指令开启事务,所以前面队列中的操作将全部取消。

我们来做个演示,首先开启一个Redis服务:

redis-server redis1.conf

连接redis:

redis-cli -h localhost -p 6379

正常执行

取消执行

因为没有执行exec,所以discard后,前面的c和d都不会执行。

组队错误 

可以看到,在组队时抱了错,事务是不会执行的。

执行错误 

我们在set c之后对c进行+1操作,由于c不是integer类型,所以报错了,但是我们发现,最终c和d被存储进了redis,从这一点来看,运行过程中的异常不会导致队列中的任务取消。

Redis和锁 

为了保证数据的安全性,我们有时候在做key的存储的时候会给key加锁,防止多个线程的情况下key被篡改去情况发生,一般发生在抢购/秒杀时对库存的操作上。

被关锁就不再多提了,使用被关锁可以完全防止key被篡改,但相应的,线程也将被阻塞,降低效率,这里说说乐观锁在redis里的使用,我们看如下操作。

首先我们需要再开一个命令行,模拟两个线程的场景:

在开始前,我们先通过flushdb指令清空redis中的数据:

监听key并开启事务

线程1:

线程2:

操作并提交key

线程1:

线程2:

可以看到线程2在对a+1的时候出问题了,返回nil。像不像CAS,没错,乐观锁的本质就是CAS,比较并交换,比较的值被改变了,自然不会存入新值。

回滚与原子性

Redis回滚

那有人要问了 ,redis到底能不能回滚?其实在上面我们已经得到了答案,就是在组队阶段是可以回滚的,看“组队错误”,在执行阶段是不支持回滚的,看“执行错误”。

所以,最终我们得出结论,在使用中,redis可支持回滚,但是情况比较复杂,是一种可预知的错误,而那些不可预知的错误,则是无法回滚。

原子性

基于以上redis事务执行时出现的组队和执行问题,所以利用redis做库存的扣减似乎是一件不太稳妥的事,所以,为了弥补这个缺陷,我们需要使用redis的原子性。原子性的特点就是:要么一起执行,要么不执行。这里我们应该都听说一个东西叫lua脚本,其特点我就不多说了,总之就是执行快且保证原子性(不被打断),但是,lua脚本也不支持回滚,所以关于redis的回滚,大家就不要纠结了。

Spring Boot接入Redis事务

关于Spring Boot接入Redis事务,怎么说呢?感觉没太大必要,哈哈,不过也不能说一无是处,用还是有人在用的,这里博主就简单讲讲怎么用。

启动Redis服务

上面Redis已经启动了,直接用。 

redisTemplate使用事务

添加依赖

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

添加配置

spring.redis.host=127.0.0.1
spring.redis.port=6379

 准备测试代码

    @Test
    void testTransaction() {
        //开启事务支持
        redisTemplate.setEnableTransactionSupport(true);
        //开启事务
        redisTemplate.multi();
        redisTemplate.opsForValue().set("name1", "codingfire1");
        redisTemplate.opsForValue().set("name2", "codingfire2");

        //执行事务
        redisTemplate.exec();
        System.out.println(redisTemplate.opsForValue().get("name1"));
        System.out.println(redisTemplate.opsForValue().get("name2"));
    }

以上代码在运行的时候发现报错了:

但是当我打开redis可视化工具查看时发现数据已经存进去了:

官方是通过Jedis来实现事务的,所以这个问题出现了还让人有点懵,很多人说是因为没开启事务支持,但是博主明显开启了,所以有些无从解答了,有知道原因的可以告诉博主。

但是活人不能让尿憋死啊,有一种SessionCallback的方式是很多人都推荐使用的,我们来看看:

    @Test
    public void testTransaction2(){
        redisTemplate.execute(new SessionCallback<List<Object>>(){
            @Override
            public List<Object> execute(RedisOperations operations) throws DataAccessException {
                operations.multi();
                operations.opsForValue().set("name3","codingfire3");
                operations.opsForValue().set("name4","codingfire4");
                operations.opsForValue().set("name5","codingfire5");
                return redisTemplate.exec();
            }
        });
    }

 执行后没报错,我们看下redis可视化工具:

测试成功。

Jedis引入

我们刚刚提到了一种Jedis实现的方式,下面我们来说说Jedis怎么实现。 

添加依赖

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

<dependency>
    <groupId>redis.clients</groupId>
    <artifactId>jedis</artifactId>
    <version>3.3.0</version>
</dependency>

添加配置

spring.redis.host=127.0.0.1
spring.redis.port=6379

 准备测试代码

    @Test
    public void testTransaction3(){
        Jedis jedis =new Jedis("127.0.0.1",6379);
        ;//开启事务
        Transaction transaction = jedis.multi();
        try {
            transaction.set("key1", "v1");
            //制造异常,测试取消事务
            //int i=1/0;
            transaction.set("key2", "v2");
            //执行事务
            transaction.exec();
        }catch (Exception e){
            //取消事务
            transaction.discard();
            e.printStackTrace();
        }
    }

查看redis可视化工具:

测试成功。 

结语

以上是对redis事务的一个实操过程,建议大家在做key的存入的时候不要对多个key进行关联,主要是避免出现数据不一致的情况。整体上就是这样,推荐使用SessionCallback和Jedis的方式,虽然你可能不一定会用到这东西,但是万一用到了呢?

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Linux基本开发工具(一)

文章目录 Linux基本开发工具&#xff08;一&#xff09;Linux安装和卸载软件Linux 软件包管理器 yum关于sudo命令关于yum源的换源问题 vim编辑器的使用vim三种模式&#xff08;常见&#xff09;vim的基本操作vim配置 Linux基本开发工具&#xff08;一&#xff09; Linux安装和…

探秘手机隐藏的望远镜功能:开启后,观察任何你想看的地方

当今的智能手机不仅仅是通信工具&#xff0c;它们蕴藏着各种隐藏的功能&#xff0c;其中之一就是让你拥有望远镜般的观察能力。是的&#xff0c;你没有听错&#xff01;今天我们将探秘手机中隐藏的望远镜功能&#xff0c;这项神奇的功能可以让你打开后&#xff0c;轻松观察任何…

/usr/bin/ld: 找不到 can‘t find -xxx++

ld是一个链接器文件&#xff0c;后面报错一般都是什么库找不到&#xff0c;so文件 我们去根目录下全局找一下该文件 cd / find -name *libstdc.so*一般都会在lib64下面找到&#xff0c;注意后面带版本号的是实际文件 我们需要把他复制到lib文件下&#xff0c;ld找的是软连接文…

3.01 用户在确认订单页收货地址操作

用户在确认订单页面&#xff0c;可以针对收货地址做如下操作&#xff1a; 1. 查询用户的所有收货地址列表 2. 新增收货地址 3. 删除收货地址 4. 修改收货地址 5. 设置默认地址步骤1&#xff1a;创建对应用户地址BO public class AddressBO {private String addressId;private…

封装上传文件组件(axios,onUploadProgress,取消请求)

目录 定时模拟进度条 方法 A.axios B.xhr 取消请求​​​​​​​ 完整代码 A.自定义上传组件 B.二次封装组件 情况 增加cancelToken不生效&#xff0c;刷新页面 进度条太快->设置浏览器网速 定时模拟进度条 startUpload() {if (!this.file) return;const totalS…

flink kafka消费者如何处理kafka主题的rebalance

背景&#xff1a; 我们日常使用kafka客户端消费kafka主题的消息时&#xff0c;当消费者退出/加入消费者组&#xff0c;kafka主题分区数有变等事件发生时&#xff0c;都会导致rebalance的发生&#xff0c;此时一般情况下&#xff0c;如果我们不自己处理offset&#xff0c;我们不…

Vue3+Vite+Pinia+Naive后台管理系统搭建之九:layout 动态路由布局

前言 如果对 vue3 的语法不熟悉的&#xff0c;可以移步Vue3.0 基础入门&#xff0c;快速入门。 1. 系统页面结构 由 menu&#xff0c;面包屑&#xff0c;用户信息&#xff0c;页面标签&#xff0c;页面内容构建 ​ 2. 创建页面 创建 src/pages/layout.vue 布局页 创建 sr…

python 封装sql 增删改查连接MySQL

select * from Teacher limit 10 连接字符串配置MysqlConfig.py class MysqlConfig:HOST 192.168.56.210PORT 3306USER rootPASSWORD 1qaz0987654321DBStudentDBCHARSET utf8封装增删改查MysqlConnection.py Author: tkhywang 2810248865qq.com Date: 2023-06-19 15:44:48 Las…

VMware Workstation及CentOS-7虚机安装

创建新的虚机&#xff1a; 选择安装软件&#xff08;这里选的是桌面版&#xff0c;也可以根据实际情况进行选择&#xff09; 等待检查软件依赖关系 选择安装位置&#xff0c;自主配置分区 ​​​​​​​ 创建一个普通用户 安装完成后重启 点击完成配置&#xff0c;进入登陆界面…

Vue3 列表渲染简单应用

去官网学习→列表渲染 | Vue.js 运行示例&#xff1a; 代码&#xff1a;HelloWorld.vue <template><div class"hello"><h1>Vue 列表渲染</h1><p v-for"item in dataList">{{item}}</p><p v-for"(item,index)…

ros tf

欢迎访问我的博客首页。 tf 1. tf 命令行工具1.1 发布 tf1.2 查看 tf 2.参考 1. tf 命令行工具 1.1 发布 tf 我们根据 cartographer_ros 的 launch 文件 backpack_2d.launch 写一个 tf.launch&#xff0c;并使用命令 roslaunch cartographer_ros tf.launch 启动。该 launch 文件…

认识所有权

专栏简介&#xff1a;本专栏作为Rust语言的入门级的文章&#xff0c;目的是为了分享关于Rust语言的编程技巧和知识。对于Rust语言&#xff0c;虽然历史没有C、和python历史悠远&#xff0c;但是它的优点可以说是非常的多&#xff0c;既继承了C运行速度&#xff0c;还拥有了Java…

zookeeper的部署

一 先下载zookeeper 二 解压包 三 修改配置文件 四 把配好文件传到其他的节点上面 五 在每个节点的dataDir指定的目录下创建一个 myid 的文件 六 配置zook的启动脚本 七 设置开机自启 八 分别启动 九查看当前状态service zookeeper status 十 总结 一 先下载zookeeper …

Vue常见的事件修饰符

1.prevent:阻止默认事件(常用) 2. stop:阻止事件冒泡(常用) 3. once:事件只触发一次(常用) 4.captrue:使用事件的捕捉模式(不常用) 5.self:只有event.target是当前操作的元素时才触发事件(不常用) 6.passive:事件的默认行为立即执行&#xff0c;无需等待事件回调执行完毕(不常用…

网关 GateWay 的使用详解、路由、过滤器、跨域配置

一、网关的基本概念 SpringCloudGateway网关是所有微服务的统一入口。 1.1 它的主要作用是&#xff1a; 反向代理&#xff08;请求的转发&#xff09; 路由和负载均衡 身份认证和权限控制 对请求限流 1.2 相比于Zuul的优势&#xff1a; SpringCloudGateway基于Spring5中…

Go微服务实践 - Rpc核心概念理解

概述 从0研究一下Golang已经Golang的微服务生态体系&#xff0c;Golang的微服务首先要从Rpc开始&#xff0c;在升级到Grpc&#xff0c;详细介绍这些技术点都在解决什么技术问题。 Rpc Rpc (Remote Procedure Call) 远程过程调用&#xff0c;简单的理解是一个节点请求另一个节…

集成学习算法是什么?如何理解集成学习?

什么是集成学习&#xff1f; 集成学习通过建立几个模型来解决单一预测问题。它的工作原理是生成多个分类器/模型&#xff0c;各自独立地学习和作出预测。这些预测最后结合成组合预测&#xff0c;因此优于任何一个单分类的做出预测。 机器学习的两个核心任务 任务一&#xff1…

【JavaEE】Spring Boot - 日志文件

【JavaEE】Spring Boot 开发要点总结&#xff08;3&#xff09; 文章目录 【JavaEE】Spring Boot 开发要点总结&#xff08;3&#xff09;1. 日志有什么作用2. 日志格式2.1 日志框架原理 3. 日志的打印3.1 System.out.println3.2 使用日志框架3.3 日志级别3.3.1 设置默认日志显…

自监督去噪:Noise2Self原理分析及实现 (Pytorch)

文章地址:https://arxiv.org/abs/1901.11365 代码地址: https://github.com/czbiohub-sf/noise2self 要点   Noise2Self方法不需要信号先验信息、噪声估计信息和干净的训练数据。唯一的假设就是噪声在测量的不同维度上表现出的统计独立性&#xff0c;而真实信号表现出一定的…

Vue Router 的query和params的区别?

区别一&#xff1a; &#xff08;1&#xff09;query相当于get请求&#xff0c;页面跳转的时候可以在地址栏看到请求参数 &#xff08;2&#xff09;params相当于post请求&#xff0c;参数不会在地址栏中显示&#xff0c;所以用params传值相对安全 &#xff08;简记&#xff1…