`

redis 事物

阅读更多
redis只能保证一个client发起的事务中的命令可以连续的执行,而中间不会插入其他client的命令。 由于redis是单线程来处理所有client的请求的所以做到这点是很容易的。

一般情况下redis在接受到一个client发来的命令后会立即处理并 返回处理结果,但是当一个client在一个连接中发出multi命令有,这个连接会进入一个事务上下文,该连接后续的命令并不是立即执行,而是先放到一 个队列中。当从此连接受到exec命令后,redis会顺序的执行队列中的所有命令。并将所有命令的运行结果打包到一起返回给client.然后此连接就 结束事务上下文。

redis> multi
OK
redis> incr a
QUEUED
redis> incr b
QUEUED
redis> exec
1. (integer) 1
2. (integer) 1
从这个例子我们可以看到incr a ,incr b命令发出后并没执行而是被放到了队列中。调用exec后俩个命令被连续的执行,最后返回的是两条命令执行后的结果


redis> multi
OK
redis> incr a
QUEUED
redis> incr b
QUEUED
redis> discard
OK
redis> get a
"1"
redis> get b
"1"
发现这次incr a incr b都没被执行。discard命令其实就是清空事务的命令队列并退出事务上下文。

虽说redis事务在本质上也相当于序列化隔离级别的了。
但是由于事务上下文的命令只排队并不立即执行,所以事务中的写操作不能依赖事务中的读操作结果

redis> multi
OK
redis> get a
QUEUED
redis> get b
QUEUED
redis> exec
1. "1"
2. "1"



redis> get a
"1"
redis> multi
OK
redis> set a 2
QUEUED
redis> exec
1. OK
redis> get a,
"2"

这样和 get a 然后直接set a是没区别的。很明显由于get a 和set a并不能保证两个命令是连续执行的(get操作不在事务上下文中)。很可能有两个client同时做这个操作。结果我们期望是加两次a从原来的1变成3. 但是很有可能两个client的get a,取到都是1,造成最终加两次结果却是2。
主要问题我们没有对共享资源a的访问进行任何的同步也就是说redis没提供任何的加锁机制来同步对a的访问。

还好redis 2.1后添加了watch命令,可以用来实现乐观锁。看个正确实现incr命令的例子,只是在前面加了watch a

redis> watch a
OK
redis> get a
"1"
redis> multi
OK
redis> set a 2
QUEUED
redis> exec
1. OK
redis> get a,
"2"
watch 命令会监视给定的key,当exec时候如果监视的key从调用watch后发生过变化,则整个事务会失败。也可以调用watch多次监视多个key.
watch的key是对整个连接有效的,事务也一样。如果连接断开,监视和事务都会被自动清除。当然了 exec,discard,unwatch命令都会清除连接中的所有监视.


redis的事务实现是如此简单,当然会存在一些问题。第一个问题是redis只能保证事务的每个命令连续执行,但是如果事务中的一个命令失败了,并不回滚其他命令
redis> set a 5
OK
redis> lpush b 5
(integer) 1
redis> set c 5
OK
redis> multi
OK
redis> incr a
QUEUED
redis> incr b
QUEUED
redis> incr c
QUEUED
redis> exec
1. (integer) 6
2. (error) ERR Operation against a key holding the wrong kind of value
3. (integer) 6

可以看到虽然incr b失败了,但是其他两个命令还是执行了。


最 后一个十分罕见的问题是 当事务的执行过程中,如果redis意外的挂了。很遗憾只有部分命令执行了,后面的也就被丢弃了。当然如果我们使用的append-only file方式持久化,redis会用单个write操作写入整个事务内容。即是是这种方式还是有可能只部分写入了事务到磁盘。发生部分写入事务的情况 下,redis重启时会检测到这种情况,然后失败退出。可以使用redis-check-aof工具进行修复,修复会删除部分写入的事务内容。修复完后就 能够重新启动了。



分享到:
评论

相关推荐

    Redis 事物(下).flv

    Redis 事物(下).flv

    redis集中处理函数封装

    该脚本文件将redis数据库处理数据的函数集中进行封装,方便使用.

    redis事务常用操作详解

    Redis 事务是数据库系统中的一种特性,它允许用户在一个操作序列中执行多个命令,这些命令要么全部执行,要么全部不执行,确保了数据的一致性和完整性。在Redis中,事务的实现主要涉及到四个核心命令:MULTI、EXEC、...

    redis面试复习.xmind

    ### 一个简单的介绍并说明了一下resdis事物的不保证原子性 # redis集群 ### redis集群三种模式 ### Redis哈希槽 ### 什么情况下会导致整个集群不可用 ### 集群三种方案之间都有哪些区别,其优点和缺点是什么 ### ...

    记录redisson实现redis分布式事务锁

    Redisson是基于Redis的Java客户端,它提供了丰富的数据结构和服务,包括分布式锁、信号量、队列、计数器等,极大地扩展了Redis在分布式系统中的应用能力。本篇文章将详细探讨如何使用Redisson实现Redis分布式事务锁...

    redis基础资料

    同时Redis还具有其它一些特性,其中包括简单的事物支持、发布订阅 ( pub/sub)、管道(pipeline)等 。 Redis具有丰富的客户端,支持现阶段流行的大多数编程语言。php常用的是phpredis, JAVA常用的是Jedis( ...

    redis配置文件

    redis缓存数据库,引入属性文件,开启事务,可以通过transcational注解控制,

    redis 使用详解

    因为redis 本事的类库 没有函数说明,使用起来很是不变,在偶然的情况下看到一篇文章,特意分享给大家 事实上,IRedisClient里面的很多方法,其实就是Redis的命令名。只要对Redis的命令熟悉一点就能够非常...事物处理

    1个Nginx+2个tomcat+2个redis服务器(主从复制)实现反向代理,实现动静分离。 - 张相逢的博客 - CSDN博客收藏20200109-134106.html

    回答好了加分,面试问到spring,你为什么用spring,spring是怎样提供事物管理的,然后spring ioc是怎么管理对象和对象之间的依赖关系的?-CSDN论坛 Spring及三大框架整合 - zgy_boke的博客 - CSDN博客 (3条消息)【二...

    Go 的 Redis 客户端.zip

    API 设计允许几乎任何事物的自定义实现。版本目前支持的基数有两个主要版本v3 是更成熟的版本,但缺少 v4 的完善 API。v3 目前仅接受错误修复。v4 具有与 v3 相同的功能,甚至更多!最大的卖点是更完善的 API。完全...

    TimeSeriesCounter:使用 Redis 作为数据库,随时间跟踪事件或事物的计数

    RedisTSCounter允许随时间跟踪事物的计数,在给定大小的桶中分割时间,并获得列表中前 N 个事物的计数排名。 聚合来自不同桶的数据时,默认行为是为所有桶分配相同的权重。 可以改用指数衰减加权,以便为最近的事件...

    Redis的Go客户端Redigo.zip

    Redigo 是 Redis 数据库的 Go 客户端。特性一个类似 Print 的 API,支持所有 Redis 命令Pipelining,包含管道事物发布/订阅连接池脚本助手类型,优化 EVALSHA 使用Helper 函数 标签:Redigo redis

    CachingFramework.Redis:基于StackExchange.Redis和Redis的分布式缓存。 包括对标记的支持并且与群集兼容

    缓存框架 基于.NET Redis客户端库添加了一些... :计算独特的事物。 :默认情况下为压缩二进制序列化器,或提供您自己的序列化。 :列表,集合,排序集,哈希和位图支持为托管集合。 Redis键空间通知:订阅发布/订

    Node.js 中清洁架构的一个用例包括 Express.js、MongoDB 和 Redis 作为主要(但可替换)的基础设施 .zip

    node.js-清洁架构Node.js 中清洁架构的一个用例包括 Express.js、MongoDB 和 Redis 作为主要(但可替换)的基础设施。概述这个示例是一个简单的 RESTful API 应用程序,用户可以通过使用Clean Architecture创建/更新...

    redis中事务机制及乐观锁的实现

     Redis目前对事物的支持相对简单。Redis只能保证一个client发起的事务中的命令可以连续的执行,而中间不会插入其他的client命令。当一个client在一个链接中发出multi命令时,这个链接会进入一个事务上下文,该连接...

    Redis教程(八):事务详解

    在事务中的所有命令都将会被串行化的顺序执行,事务执行期间,Redis不会再为其它客户端的请求提供任何服务,从而保证了事物中的所有命令被原子的执行。  2). 和关系型数据库中的事务相比,在Redis事务中如果有某一...

    zppt5706.zip

    6. **事务**:虽然不支持传统数据库的ACID特性,但Redis提供了一种简单的事物模型,可以一次性执行多个命令。 7. **主从复制**:通过复制机制,可以创建多个从节点,提高读取性能和容错能力。 8. **Sentinel监控与...

    简单注解实现集群同步锁(spring+redis+注解)

    1. **配置Redis连接**:首先需要在项目中集成Spring Data Redis,配置Redis连接信息,以便应用能够与Redis服务器通信。 2. **创建自定义切面**:创建一个名为`RedisLockAspect`的切面,该切面会在运行时检查带有`@...

Global site tag (gtag.js) - Google Analytics