四.redis 事务
redis对事务的支持目前还比较简单。redis只能保证一个client发起的事务中的命令可以连续的执行,而中间不会插入其他client的命令。 由于redis是单线程来处理所有client的请求的所以做到这点是很容易的。一般情况下redis在接受到一个client发来的命令后会立即处理并 返回处理结果,但是当一个client在一个连接中发出multi命令有,这个连接会进入一个事务上下文,该连接后续的命令并不是立即执行,而是先放到一 个队列中。当从此连接受到exec命令后,redis会顺序的执行队列中的所有命令。并将所有命令的运行结果打包到一起返回给client.然后此连接就 结束事务上下文。下面可以看一个例子
redis> multiOKredis> incr aQUEUEDredis> incr bQUEUEDredis> exec1. (integer) 12. (integer) 1
从这个例子我们可以看到incr a ,incr b命令发出后并没执行而是被放到了队列中。调用exec后俩个命令被连续的执行,最后返回的是两条命令执行后的结果
我们可以调用discard命令来取消一个事务。接着上面例子
redis> multiOKredis> incr aQUEUEDredis> incr bQUEUEDredis> discardOKredis> get a"1"redis> get b"1"可以发现这次incr a incr b都没被执行。discard命令其实就是清空事务的命令队列并退出事务上下文。
redis> multiOKredis> get aQUEUEDredis> get bQUEUEDredis> exec1. "1"2. "1"发现问题了吧。假如我们想用事务实现incr操作怎么办?可以这样做吗?
redis> get a"1"redis> multiOKredis> set a 2QUEUEDredis> exec1. OKredis> get a,"2"结论很明显这样是不行的。这样和 get a 然后直接set a是没区别的。很明显由于get a 和set a并不能保证两个命令是连续执行的(get操作不在事务上下文中)。很可能有两个client同时做这个操作。结果我们期望是加两次a从原来的1变成3. 但是很有可能两个client的get a,取到都是1,造成最终加两次结果却是2。主要问题我们没有对共享资源a的访问进行任何的同步
redis> watch aOKredis> get a"1"redis> multiOKredis> set a 2QUEUEDredis> exec1. OKredis> get a,"2"watch 命令会监视给定的key,当exec时候如果监视的key从调用watch后发生过变化,则整个事务会失败。也可以调用watch多次监视多个key.这 样就可以对指定的key加乐观锁了。注意watch的key是对整个连接有效的,事务也一样。如果连接断开,监视和事务都会被自动清除。当然了 exec,discard,unwatch命令都会清除连接中的所有监视.
redis> set a 5OKredis> lpush b 5(integer) 1redis> set c 5OKredis> multiOKredis> incr aQUEUEDredis> incr bQUEUEDredis> incr cQUEUEDredis> exec1. (integer) 62. (error) ERR Operation against a key holding the wrong kind of value3. (integer) 6可以看到虽然incr b失败了,但是其他两个命令还是执行了。
?
from:http://www.cnblogs.com/xhan/archive/2011/02/04/1949151.html