`
longzhun
  • 浏览: 371949 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

四.redis 事务

 
阅读更多

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后俩个命令被连续的执行,最后返回的是两条命令执行后的结果
我们可以调用discard命令来取消一个事务。接着上面例子

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"
发现问题了吧。假如我们想用事务实现incr操作怎么办?可以这样做吗?
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.这 样就可以对指定的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工具进行修复,修复会删除部分写入的事务内容。修复完后就 能够重新启动了。

 

from:http://www.cnblogs.com/xhan/archive/2011/02/04/1949151.html

分享到:
评论

相关推荐

    ServiceStack.Redis-5.8无限制.zip

    5. **事务处理**:Redis支持单个命令的原子执行,ServiceStack.Redis通过`RedisTransaction`类提供了事务处理功能,允许一组命令作为一个整体执行,确保数据的一致性。 6. **发布/订阅(Pub/Sub)**:ServiceStack....

    StackExchange.Redis .NET4.0

    7. **事务**:Redis支持事务,StackExchange.Redis通过`Multi`、`Execute`和`Discard`方法来包裹一系列操作,确保它们作为一个原子单元执行。 8. **脚本支持**:Redis允许使用Lua脚本来执行一连串操作,...

    ServiceStack.Redis操作工具类

    5. **事务(Transactions)**:Redis支持事务,ServiceStack.Redis提供了`IRedisTransaction`接口来处理。事务可以批量执行命令,并且在所有命令都成功执行或全部失败的情况下返回结果。 6. **脚本(Lua Scripting...

    ServiceStack.redis v3.9.71

    此外,它还支持事务、持久化和主从复制等Redis特性。 StackExchange.Redis.1.2.6是另一个Redis客户端库,尽管它不是ServiceStack.redis的一部分,但常常与ServiceStack.redis一起使用。StackExchange.Redis提供了更...

    StackExchange.Redis-1.2.6精简,附64位dll

    4. **事务支持**:Redis支持事务操作,StackExchange.Redis通过`Transaction`类提供事务API,可以确保一组操作的原子性,即所有操作要么全部成功,要么全部失败。 5. **异步编程**:为了适应现代高性能Web应用,...

    ServiceStack.Redis 3.9

    5. **事务支持**:ServiceStack.Redis允许用户在Redis中执行原子的事务操作,确保数据的一致性。 6. **序列化**:库内置了对多种序列化器的支持,包括Json.NET、protobuf等,这使得在Redis中存储和检索对象变得容易...

    C# 使用 ServiceStack.Redis 必须的4个dll

    ServiceStack.Redis还支持事务、脚本执行和分布式锁等功能,大大增强了在.NET环境下的Redis操作能力。 博客链接中的文章(https://blog.csdn.net/u010067685/article/details/100122312)可能提供了更具体的使用...

    StackExchange.Redis-1.2.6

    StackExchange.Redis是一个强大的C#客户端库,用于与Redis内存数据存储进行交互。Redis是一个高性能的键值对数据库,广泛应用于缓存、消息队列、事件发布/订阅等多种场景。版本1.2.6是该库的一个稳定版本,包含了...

    (5.9.3.0)ServiceStack.Redis(已取消6000限制)

    8. 事务处理:使用`BeginTransaction`、`Execute`和`Commit`进行多条命令的原子操作。 此外,ServiceStack.Redis还支持高级特性,如Lua脚本执行、主从复制配置、持久化策略设置等。这个特定的5.9.3.0版本由于取消了...

    StackExchange.Redis-中文使用文档.pdf

    9. 事务:StackExchange.Redis中文使用文档中还介绍了事务的相关知识点,包括如何使用ConnectionMultiplexer类来实现事务。 StackExchange.Redis中文使用文档是Redis C# 客户端的详细使用指南,涵盖了基础配置、...

    StackExchange.Redis文档翻译.pdf

    StackExchange.Redis支持Redis的事务功能,允许在一个操作中执行多条命令,并确保它们原子性地执行。 6. **事件**: 库提供了事件处理机制,如`SubscriptionChanged`,可以在订阅或取消订阅时触发回调。 7. **...

    ServiceStack.Redis.dll

    4. **事务(Transactions)**:支持原子操作的事务处理,确保多个命令的执行顺序和一致性。 5. **Lua脚本支持**:利用Redis的内置Lua引擎,可以执行复杂的脚本操作,提高性能并减少网络通信。 6. **持久化**:...

    ServiceStack.Redis 源码

    6. **事务处理**:支持 Redis 事务,允许在一个原子操作中执行多个命令,确保数据的一致性。 7. **Lua 脚本支持**:利用 Redis 的 Lua 脚本执行能力,可以在服务器端执行复杂的逻辑,减少网络通信次数。 8. **持久化...

    c#操作redis驱动ServiceStack.Redis,V3.9版本,免费不限制次数免破解

    - **事务(Transactions)**:使用`redis.Transaction.Do Transactions(transactions => {...})`执行多条命令的原子操作。 - **脚本(Lua Scripting)**:通过`redis.ExecuteLuaScript`方法执行Redis的内置Lua脚本...

    stackexchange.redis.2.0.513.nupkg

    3. **事务处理**:StackExchange.Redis支持Redis的事务功能,允许用户在单个操作中执行多个命令,保证原子性。 4. **发布/订阅**:提供Pub/Sub(发布/订阅)模式,实现消息传递和事件驱动架构。 5. **命令缓冲与...

    ServiceStack.Redis.3.9.29.0

    5. **事务支持**:支持Redis的事务操作,允许在单个命令中执行多个操作,确保数据一致性。 6. **连接池管理**:通过连接池来复用Redis连接,提高性能并降低资源消耗。 7. **强大的Redis客户端管理**:提供了管理多个...

    ServiceStack.Redis客户端dLL3.0

    除了基本的数据操作,ServiceStack.Redis还支持高级特性,如持久化、事务处理、Lua脚本执行等。Lua脚本可以在服务器端执行,减少网络往返,提高性能。此外,客户端还提供了连接池管理,通过复用连接降低建立和关闭...

    Another.Redis.Desktop.Manager.1.3.5.zip

    除了基本的数据操作,该工具还具有命令行执行功能,用户可以直接输入Redis命令,进行高级操作,如发布/订阅消息、执行事务等。这对于开发者来说,无疑大大提升了调试和测试的便利性。同时,它还支持导出和导入数据,...

    StackExchange.Redis

    6. **事务处理**:Redis支持事务,StackExchange.Redis提供了相应的API,可以将一组命令封装在一个事务中,确保它们要么全部成功,要么全部失败,提供原子性保障。 7. **脚本支持**:通过使用`Eval`和`EvalSHA`方法...

    StackExchange.Redis.dll(1.2.6)

    6. **事务处理**:支持Redis的事务操作,允许在一次原子操作中执行多个命令。 7. **兼容性**:除了基本的Redis命令,StackExchange.Redis还提供了对Redis特定扩展特性的支持,如HyperLogLog、Geo、Stream等。 8. *...

Global site tag (gtag.js) - Google Analytics