`
- 浏览:
165283 次
- 性别:
- 来自:
北京
-
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工具进行修复,修复会删除部分写入的事务内容。修复完后就 能够重新启动了。
分享到:
Global site tag (gtag.js) - Google Analytics
相关推荐
Redis 事物(下).flv
该脚本文件将redis数据库处理数据的函数集中进行封装,方便使用.
Redis 事务是数据库系统中的一种特性,它允许用户在一个操作序列中执行多个命令,这些命令要么全部执行,要么全部不执行,确保了数据的一致性和完整性。在Redis中,事务的实现主要涉及到四个核心命令:MULTI、EXEC、...
### 一个简单的介绍并说明了一下resdis事物的不保证原子性 # redis集群 ### redis集群三种模式 ### Redis哈希槽 ### 什么情况下会导致整个集群不可用 ### 集群三种方案之间都有哪些区别,其优点和缺点是什么 ### ...
Redisson是基于Redis的Java客户端,它提供了丰富的数据结构和服务,包括分布式锁、信号量、队列、计数器等,极大地扩展了Redis在分布式系统中的应用能力。本篇文章将详细探讨如何使用Redisson实现Redis分布式事务锁...
同时Redis还具有其它一些特性,其中包括简单的事物支持、发布订阅 ( pub/sub)、管道(pipeline)等 。 Redis具有丰富的客户端,支持现阶段流行的大多数编程语言。php常用的是phpredis, JAVA常用的是Jedis( ...
redis缓存数据库,引入属性文件,开启事务,可以通过transcational注解控制,
因为redis 本事的类库 没有函数说明,使用起来很是不变,在偶然的情况下看到一篇文章,特意分享给大家 事实上,IRedisClient里面的很多方法,其实就是Redis的命令名。只要对Redis的命令熟悉一点就能够非常...事物处理
回答好了加分,面试问到spring,你为什么用spring,spring是怎样提供事物管理的,然后spring ioc是怎么管理对象和对象之间的依赖关系的?-CSDN论坛 Spring及三大框架整合 - zgy_boke的博客 - CSDN博客 (3条消息)【二...
API 设计允许几乎任何事物的自定义实现。版本目前支持的基数有两个主要版本v3 是更成熟的版本,但缺少 v4 的完善 API。v3 目前仅接受错误修复。v4 具有与 v3 相同的功能,甚至更多!最大的卖点是更完善的 API。完全...
RedisTSCounter允许随时间跟踪事物的计数,在给定大小的桶中分割时间,并获得列表中前 N 个事物的计数排名。 聚合来自不同桶的数据时,默认行为是为所有桶分配相同的权重。 可以改用指数衰减加权,以便为最近的事件...
Redigo 是 Redis 数据库的 Go 客户端。特性一个类似 Print 的 API,支持所有 Redis 命令Pipelining,包含管道事物发布/订阅连接池脚本助手类型,优化 EVALSHA 使用Helper 函数 标签:Redigo redis
缓存框架 基于.NET Redis客户端库添加了一些... :计算独特的事物。 :默认情况下为压缩二进制序列化器,或提供您自己的序列化。 :列表,集合,排序集,哈希和位图支持为托管集合。 Redis键空间通知:订阅发布/订
node.js-清洁架构Node.js 中清洁架构的一个用例包括 Express.js、MongoDB 和 Redis 作为主要(但可替换)的基础设施。概述这个示例是一个简单的 RESTful API 应用程序,用户可以通过使用Clean Architecture创建/更新...
Redis目前对事物的支持相对简单。Redis只能保证一个client发起的事务中的命令可以连续的执行,而中间不会插入其他的client命令。当一个client在一个链接中发出multi命令时,这个链接会进入一个事务上下文,该连接...
在事务中的所有命令都将会被串行化的顺序执行,事务执行期间,Redis不会再为其它客户端的请求提供任何服务,从而保证了事物中的所有命令被原子的执行。 2). 和关系型数据库中的事务相比,在Redis事务中如果有某一...
6. **事务**:虽然不支持传统数据库的ACID特性,但Redis提供了一种简单的事物模型,可以一次性执行多个命令。 7. **主从复制**:通过复制机制,可以创建多个从节点,提高读取性能和容错能力。 8. **Sentinel监控与...
1. **配置Redis连接**:首先需要在项目中集成Spring Data Redis,配置Redis连接信息,以便应用能够与Redis服务器通信。 2. **创建自定义切面**:创建一个名为`RedisLockAspect`的切面,该切面会在运行时检查带有`@...