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

Redis--事务

目录

一、事务介绍

二、事务操作

2.1 MULTI

2.2 EXEC

2.3 DISCARD

2.4 WATCH

2.5 UNWATCH


一、事务介绍

Redis 的事务和 MySQL 的事务概念上是类似的. 都是把一系列操作绑定成⼀组. 让这⼀组能够批量执行.
但是注意体会 Redis 的事务和 MySQL 事务的区别:
       1.弱化的原子性:Redis没有“回滚机制”,只能做到这些操作“批量进行”。不能做到“一个失败就回到最初状态”。
       2.不保证一致性:不涉及“约束”。也没有回滚。MySQL的一致性体现的是运行事务前与运行事务后,结果都是合理有效的,不会出现中间非法状态。
       3.不需要隔离性:也没有隔离级别,因为不会并发处理事务(Redis单线程处理请求)
       4.不需要持久性:是保存在内存的,是否开启持久化,是redis-server自己的事情,和事务无关。
Redis事务的本质是在服务器上搞了一个“事务队列”。每次客户端在事务中进行一个操作,都会把命令先发给服务器,放到“事务队列”中(但是并不会立即执行)而是在真正受到EXEC命令之后,才真正执行队列中所有操作
因此, Redis 的事务的功能相比于 MySQL 来说, 是弱化很多的. 只能保证事务中的这几个操作是 "连续的", 不会被别的客户端 "加塞", 仅此而已.

二、事务操作

2.1 MULTI

开启一个事务,执行成功返回OK。

127.0.0.1:6379> MULTI
OK

2.2 EXEC

真正执行事务。

示例:

127.0.0.1:6379> MULTI
OK
127.0.0.1:6379> set k1 1
QUEUED
127.0.0.1:6379> set k2 2
QUEUED
127.0.0.1:6379> set k3 3
QUEUED
127.0.0.1:6379> EXEC
1) OK
2) OK
3) OK
每次添加⼀个操作, 都会提示 "QUEUED", 说明命令已经进⼊客户端的队列了.
真正执行EXEC 的时候, 客户端才会真正把上述操作发送给服务器.
此时就可以获取到上述 key 的值了
127.0.0.1:6379> get k1
"1"
127.0.0.1:6379> get k2
"2"
127.0.0.1:6379> get k3
"3"

2.3 DISCARD

放弃当前事务. 此时直接清空事务队列. 之前的操作都不会真正执行到.
示例:
 127.0.0.1:6379> MULTIOK127.0.0.1:6379> set k1 1QUEUED127.0.0.1:6379> set k2 2QUEUED127.0.0.1:6379> DISCARDOK127.0.0.1:6379> get k1(nil)127.0.0.1:6379> get k2(nil)

2.4 WATCH

在执行事务的时候, 如果某个事务中修改的值, 被别的客户端修改了, 此时就容易出现数据不⼀致的问题.
示例:
# 客⼾端1 先执⾏127.0.0.1:6379> MULTIOK127.0.0.1:6379> set key 100QUEUED# 客⼾端2 再执⾏127.0.0.1:6379> set key 200OK
# 客⼾端1 最后执⾏127.0.0.1:6379> EXEC1) OK
此时, key 的值是多少呢??
从输⼊命令的时间看, 是客户端1 先执行的 set key 100. 客户端2 后执行的 set key 200.
但是从实际的执行时间看, 是客户端2 先执行的, 客户端1 后执行的.
127.0.0.1:6379> get key 
"100"
这个时候, 其实就容易引起歧义.
因此, 即使不保证严格的隔离性, 至少也要告诉用户, 当前的操作可能存在风险
watch命令就是用来解决这个问题的。watch在该客户端上监控一组具体的key
    当开启事务的时候,如果对watch的key进行修改,就会记录下来当前key的“版本号”。(版本号是个简单的整数, 每次修改都会使版本变大. 服务器来维护每个 key 的版本号情况)
    
     在真正提交事务的时候, 如果发现当前服务器上的 key 的版本号已经超过了事务开始时的版本号, 就会让事务执行失败. (事务中所有的操作都不执行)
示例:
客户端一先执行
127.0.0.1:6379> watch k1 # 开始监控 k1
OK
127.0.0.1:6379> MULTI
OK
127.0.0.1:6379> set k1 100 # 进⾏修改, 从服务器获取 k1 的版本号是 0. 记录 k1 的
版本号. (还没真修改呢, 版本号不变)
QUEUED
127.0.0.1:6379> set k2 1000 
QUEUED

只是入队列,但不执行事务

客户端二再执行

127.0.0.1:6379> set k1 200 # 修改成功, 使服务器端的 k1 的版本号 0 -> 1
OK

客户端一再执行

127.0.0.1:6379> EXEC # 真正执⾏修改操作, 此时对⽐版本发现, 客⼾端的 k1 的版
本号是 0, 服务器上的版本号是 1, 版本不⼀致! 说明有其他客⼾端在事务中间修改了 k1 !!! 
(nil)
127.0.0.1:6379> get k1
"200"
127.0.0.1:6379> get k2
(nil)

此时说明事务已经被取消了,这次提交的所以命令都没执行

2.5 UNWATCH

取消对 key 的监控.
相当于 WATCH 的逆操作,此处不做演示
http://www.xdnf.cn/news/32.html

相关文章:

  • 从彩色打印单行标准九九表学习〖代码情书〗的书写范式(Python/DeepSeek)
  • 总结【过往部分项目经历一(计算机图形学方向)】
  • 一路磕磕绊绊解决flutter doctor 报错CocoaPods not installed
  • 鸿蒙API15 “一多开发”适配:解锁黄金三角法则,开启高效开发新旅程
  • Gateway
  • 【HDFS入门】HDFS高可用性与容错机制深度解析
  • XC7K410T‑2FFG900I 赛灵思XilinxFPGA Kintex‑7
  • 5.VTK 相机
  • 智能体开发范式革命:Cangjie Magic的颠覆性创新与行业重塑
  • 电控---printf重定向输出
  • BFC详解
  • 4.16 AT好题选做
  • 2026《数据结构》考研复习笔记二(C++面向对象)
  • QML 信号与槽
  • 微信小程序文字混合、填充动画有效果图
  • 全自动驾驶(FSD,Full Self-Driving)自动驾驶热点技术的成熟之处就是能判断道路修复修路,能自动利用类似“人眼”的摄像头进行驾驶!值得学习!
  • SpringBoot项目动态加载jar 实战级别
  • 探索鸡养殖虚拟仿真实验:科技赋能养殖新体验
  • 新型多机器人协作运输系统,轻松应对复杂路面
  • 黑马商城项目(三)微服务
  • IDEA 中 Scala 项目远程连接虚拟机 Spark 环境
  • ubuntu 向右拖动窗口后消失了、找不到了
  • Nodemon vs. PM2:开发与生产环境的 Node.js 部署最佳实践
  • 【FFmpeg从入门到精通】第二章-FFmpeg工具使用基础
  • 数据通信学习笔记之OSPF路由汇总
  • ThingsBoard3.9.1 MQTT Topic(2)
  • iptables防火墙
  • NO.96十六届蓝桥杯备战|图论基础-多源最短路|Floyd|Clear And Present Danger|灾后重建|无向图的最小环问题(C++)
  • Doris FE 常见问题与处理指南
  • 告别昂贵语音合成服务!用GPT-SoVITS生成你的个性化AI语音