深入学习 Redis - 事务、实现原理、指令使用及场景

目录

一、Redis 事务 vs MySQL事务

二、Redis 事务的执行原理

2.1、执行原理

2.2、Redis 事务设计这么简单,为什么不涉及成 MySQL 那样强大呢?

三、Redis 事务的使用

3.1、使用场景

3.2、具体演示

开启/执行/放弃事务

watch 监控

watch 实现原理

小结


一、Redis 事务 vs MySQL事务


我们熟知的 MySQL,你还记得她的事务特性么?

  1. 原子性:将多个操作打包成一个整体,要么全都执行成功,要么一个都不执行,一旦执行出错,立刻回滚如初.
  2. 一致性:事务执行前后,通过约束和回滚机制,保证数据合理. 例如我去银行给 张三 转账 1 元钱,那么我的卡里就会减少 1 元,张三的卡里就会增加 1 元,不能增加 100 元。
  3. 持久性:事务做出的修改都会存储到硬盘上,不会随着服务器重启而丢失.
  4. 隔离性:事务并发执行,涉及到的一些问题.

Redis 的事务和 MySQL 相比,就像一个“弟弟”~

  1. 原子性:Redis 的事务到底有没有在原子性,存在争议!因为从 Redis 的角度理解就是“把多个操作打包到一起,要么全都执行,要么全都不执行”,也就是说,这里如果中途执行失败,那就失败吧,不会有像 MySQL 那样回滚的操作(也因此,网上有人一部分人说 redis 事务有原子性,一部分说没有原子性,都对,但是要看从那个角度出发了~).
  2. 不具备一致性:redis 没有约束和回滚,事务执行一旦出错,就可能导致不一致的情况.
  3. 不具备持久性:Redis 本身就是内存数据库,数据是存储在内存中的. 虽然 Redis 也有持久化机制,但是这里的持久化机制和事务没有什么直接关系.
  4. 不具备隔离性:Redis 是一个单线程模型的服务器程序,所有的请求 / 事务,都是 “串行” 执行的.

二、Redis 事务的执行原理


2.1、执行原理

Redis 事务的主要意义就是为了 “打包” ,避免其他客户端的命令,插队到其中,那他具体是通过什么实现的呢?Redis 实现事务,是引入了队列(每一个客户端都有),开启事务的时候,客户端输入的命令,就会发给服务器,然后进入都这个队列中,此时并不会立即执行,当收到 “执行事务” 命令的时候,就会把队列中的这些任务按照顺序依次执行(这里是由 Redis 主线程执行的,他会把事务中的操作执行完,再去执行其他客户端).

举个例子,这就像是你去做核酸检测,但是核酸检测是早上 7 点开始,但是你 6 点就来了,还发现已经有一些人在这里排成长队,所以这个时候你只能在去后面排队,不能插队,等到核酸检测开始,并且前面的人都检测完了,才轮到你~

2.2、Redis 事务设计这么简单,为什么不涉及成 MySQL 那样强大呢?

MySQL 的事务之所以能设计的这么强大(原子性、一致性....),背后是付出了很多代价的~

  1. 空间上,要花费跟多的空间来存储更多的数据,例如 MySQL 事务执行中途出错,引发的回滚机制,这种回滚机制也是需要花费空间去记录每条执行的指令是什么,才能进行回滚的.
  2. 时间上,也要有更大的执行开销,比如 MySQL 事务的持久化机制,是存储在硬盘上的,操作缓慢;再例如 MySQL 中事务遇到冲突是进行加锁,加锁也是需要开销的,涉及到用户态到内核态的转化,而且加锁就有可能产生锁竞争,一竞争就会加锁排队,一排队就不知道什么时候释放锁了~

也正因如此,Redis 才有了上场的机会。

三、Redis 事务的使用


3.1、使用场景

如果需要把多个操作打包进行,使用事务就是比较合适的~

比如说以前春运,咱们在网上抢火车票的场景、秒杀...

这里以抢火车票的场景为例:

假设现在有大量的用户都在同一时间段开始抢票,也就是说有多个 Redis 客户端执行执行抢票操作,如下:

获取仓库中剩余的票数;
if(剩余的票数 > 0) {下单成功;商品数量--;
}

这里如果不加任何限制,就有可能引发 “线程安全” 问题~  以前咱们解决线程安全问题,都是通过加锁排队来避免 “插队” 的,而 redis 这里不加锁,也能解决上述问题,直接使用事务即可,如下:

开启事务
get count
if count > 0decr count
执行事务

Ps:Redis 原生的命令中是没有上述这种判断的,但是 Redis 支持 lua 脚本,通过 lua 脚本就可以实现上述的 条件判定,并且也和事务一样是打包批量执行的.

确实,redis 的事务使用场景没有 mysql 事务那么多,并且 redis 如果是按照集群模式部署,是不支持事务的.

3.2、具体演示

redis 的事务是通过以下命令进行控制的

multi        开启事务(读作 “猫体”)
exec         执行事务
discard      放弃当前事务
watch        监控某个 key 是否在事务执行之前发生变化(必须搭配事务使用)
unwatch      放弃监控

开启/执行/放弃事务

开启事务,执行以下命令:

此时开启另一个客户端,查看这几个 key ,会发现这几个 key 并没有被赋值(说明此时还没有执行事务).,如下

如果放弃事务,就相当于什么也没有发生,如下

如果使用 exec,就会按顺序执行事务,如下

watch 监控

watch 就是用来监控某个 key 是否在事务执行之前,发生改变,但必须搭配事务来使用.

如下,用 watch 监控 key1,开启事务,并在执行事务之前,另一个客户端对 key1 进行修改,如下

执行事务后,发现 key 在外部由修改,会返回 nil ,表示事务什么都不会执行,如下

watch 实现原理

watch 的实现,类似于一个 “乐观锁”.

乐观锁,不是某个具体的锁,而是指某一类锁的特性:加锁之前,就会有一个心里预期,预期接下来锁冲突的概论比较低,就像是有些同学,考试前觉得自己平时学的很扎实,因此考试前就轻松一些,少复习一点~

redis 的 watch 就相当于基于 “版本号” 这样的机制,实现 “乐观锁”。

当执行 watch key 的时候,就会给 key 安排一个 版本号,版本号可以理解成一个“整数”,每次在修改 key 的时候,版本号都会 “变大” (这个变大是没有规律的,不是每次都增长1),然后在执行 事务 的时候,就会做出判定,判断当前这个 key 的版本号和最初 watch 的时候,记录的版本号是否一致~

  1. 如果一致,说明当前 key 在事务开启到最终执行这个过程中,没有别的客户端修改,才能真正的执行事务.
  2. 如果不一致,就说明 key 在其他客户端修改过了,因此就直接丢弃事务中的所有操作,最后返回 nil.

Ps:这样的设定,在 CAS 的 ABA 问题中也涉及到过,思想方法和实现上都是非常相似的,例如 CAS 实现自旋锁,就类似版本号设定的实现.

小结


重在学习思想,而不是理论,很多思想都是触类旁通的,当我们未来遇到某一类问题的时候,就可以回想起当时解决问题的思想方法,迎刃而解~

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

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

相关文章

chapter14:springboot与安全

Spring Boot与安全视频 Spring Security, shiro等安全框架。主要功能是”认证“和”授权“,或者说是访问控制。 认证(Authentication)是建立在一个声明主体的过程(一个主体一般指用户,设备或一些可以在你的应用程序中…

Django之JWT库与SimpleJWT库的使用

Django之JWT库与SimpleJWT库的使用 JWTJWT概述头部(header)载荷(payload)签名(signature) Django使用JWT说明jwt库的使用安装依赖库配置settings.py文件配置urls.py文件创建视图配置权限 SimpleJWT库的使用安装SimpleJWT库配置Django项目配置路由创建用户接口测试身份认证自定义…

【雕爷学编程】Arduino动手做(190)---MAX4466声音模块

37款传感器与模块的提法,在网络上广泛流传,其实Arduino能够兼容的传感器模块肯定是不止37种的。鉴于本人手头积累了一些传感器和执行器模块,依照实践出真知(一定要动手做)的理念,以学习和交流为目的&#x…

Jenkins Gerrit Trigger实践

1.创建Gerrit Trigger 2.jenkins master节点生成gerrit用户的密钥 这里的用户名得写登录gerrit后个人信息中的 Username 3.gerrit 配置刚刚jenkins生成密钥的公钥 4.gerrit 用户加入群组 不加这个群组,下一步测试就会报错“User aeshare has no capability conn…

Dueling Network

Dueling Network —— Dueling Network Architectures for Deep Reinforcement Learning 论文下载地址 论文介绍 图9. Dueling Network 模型结果示意图 Dueling Network与传统DQN的区别在于神经网络结构的不同,Dueling Netowrk在传统DQN的基础上只进行了微小的改动…

Flowise AI:用于构建LLM流的拖放UI

推荐:使用NSDT场景编辑器助你快速搭建可二次编辑的3D应用场景 什么是Flowise AI? Flowise AI是一个开源的UI可视化工具,用于帮助开发LangChain应用程序。在我们详细介绍 Flowise AI 之前,让我们快速定义 LangChain。LangChain是…

MapTR论文笔记

MAPTR: STRUCTURED MODELING AND LEARNING FOR ONLINE VECTORIZED HD MAP CONSTRUCTION 目的 传统高精地图 通过一些离线的基于 SLAM 的方法生成,需要复杂的流程以及高昂的维护费用。基于 bev 分割的建图方法,缺少向量化 实例级的信息,比如…

基于Vue+wangeditor实现富文本编辑

目录 前言分析实现具体解决的问题有具体代码实现如下效果图总结前言 一个网站需要富文本编辑器功能的原因有很多,以下是一些常见的原因: 方便用户编辑内容:富文本编辑器提供了类似于Office Word的编辑功能,使得那些不太懂HTML的用户也能够方便地编辑网站内容。提高用户体验…

从零开始实现一个 mini-Retrofit 框架

前言 本篇文章将采用循序渐进的编码方式,从零开始实现一个Retorift框架,在实现过程中不断提出问题并分析实现,最终开发出一个mini版的Retrofit框架 演示一个使用OkHttp的项目Demo 为了更好的演示框架的实现过程,这里我先创建了一…

Compose应用案例(利用docker compose安装lnmp实例)

目录 Compose应用案例 一、前提配置 (一)安装docker-ce(Linux安装Docker) (二)安装docker-compose 二、安装docker compose部署lnmp (一)目录结构: (二…

SPM(Swift Package Manager)开发及常见事项

SPM怎么使用的不再赘述,其优点是Cocoapods这样的远古产物难以望其项背的,而且最重要的是可二进制化、对xcproj项目无侵入,除了网络之外简直就是为团队开发的项目库依赖最好的管理工具,是时候抛弃繁杂低下的cocoapods了。 一&…

c语言——杨辉三角

//杨辉三角 #include<stdio.h> int main() {int i,j,k,n0,a[10][10];while(n<0||n>13){/*行数不超过13&#xff0c;为了显示规范*/printf("n即输入行数");scanf("%d",&n);}printf("%d行杨辉三角如下&#xff1a;\n",n);for(i1;i…

SpringCloud(29):Nacos简介

1 什么是配置中心 1.1 什么是配置 应用程序在启动和运行的时候往往需要读取一些配置信息&#xff0c;配置基本上伴随着应用程序的整个生命周期&#xff0c;比如&#xff1a;数据库连接参数、启动参数等。 配置主要有以下几个特点&#xff1a; 配置是独立于程序的只读变量 …

IPsec

数据认证 数据认证是指验证数据的真实性、完整性和可信度的过程。它确保数据在传输、存储和处理过程中没有被篡改或损坏&#xff0c;并且数据的来源可信。 数据认证的主要作用包括&#xff1a; 数据完整性保护&#xff1a;通过校验和、哈希算法等方法&#xff0c;检测数据是否…

GoogLeNet卷积神经网络-笔记

GoogLeNet卷积神经网络-笔记 GoogLeNet是2014年ImageNet比赛的冠军&#xff0c; 它的主要特点是网络不仅有深度&#xff0c; 还在横向上具有“宽度”。 由于图像信息在空间尺寸上的巨大差异&#xff0c; 如何选择合适的卷积核来提取特征就显得比较困难了。 空间分布范围更广的…

On Evaluation of Embodied Navigation Agents 论文阅读

论文信息 题目&#xff1a;On Evaluation of Embodied Navigation Agents 作者&#xff1a;Peter Anderson&#xff0c;Angel Chang 来源&#xff1a;arXiv 时间&#xff1a;2018 Abstract 过去两年&#xff0c;导航方面的创造性工作激增。这种创造性的输出产生了大量有时不…

MyBatis快速入门

MyBatis快速入门 MyBatis简介 什么是MyBatis? MyBatis 是一款优秀的持久层框架&#xff0c;用于简化JDBC开发MyBatis本是Apache 的一个开源项目iBatis,2010年这个项目由apache softwarefoundation 迁移到了google code&#xff0c;并且改名为MyBatis 2013年11月迁移到Githu…

如何推导椭圆的参数方程

椭圆基础知识 椭圆定义&#xff1a;椭圆上任意一点到两焦点的距离之和为2a 如何由椭圆定义推出椭圆标准方程呢&#xff1f; 如上图所示。 由定义可得已知条件为 ∣ M C 1 ∣ ∣ M C 2 ∣ 2 a 当 M 落在顶点 P 上时&#xff0c;可得另一已知条件 a 2 − b 2 c 2 当有了已…

unity行为决策树实战详解

一、行为决策树的概念 行为决策树是一种用于游戏AI的决策模型&#xff0c;它将游戏AI的行为分解为一系列的决策节点&#xff0c;并通过节点之间的连接关系来描述游戏AI的行为逻辑。在行为决策树中&#xff0c;每个节点都代表一个行为或决策&#xff0c;例如移动、攻击、逃跑等…

SpringBoot3 整合Prometheus + Grafana

通过Prometheus Grafana对线上应用进行观测、监控、预警… 健康状况【组件状态、存活状态】Health运行指标【cpu、内存、垃圾回收、吞吐量、响应成功率…】Metrics… 1. SpringBoot Actuator 1. 基本使用 1. 场景引入 <dependency><groupId>org.springframew…