redis事务详解,带你搞懂redis的事务

104 阅读4分钟

文章目录

官方文档

redis.io/docs/manual…

一、为什么要用事务

Redis的单个命令是原子性的(比如 get 、set、mget、mset),要么成功要么失败,不存在并发干扰的问题。

如果涉及到多个命令的时候,需要把多个命令作为一个不可分割的处理序列,就必须要依赖Redis的功能特性来实现了。

Redis提供了事务的功能,可以把一组命令一起执行。Redis的事务有3个特点:

  • 按进入队列的顺序执行。
  • 不会受到其他客户端的请求的影响。
  • 事务不能嵌套,多个multi命令效果一样。

二、事务的用法

Redis的事务涉及到四个命令:multi(开启事务),exec(执行事务),discard(取消事务),watch(监视)。

1、使用实例

tom和mic各有1000元,tom向mic转账100元。

127.0.0.1:6379> set tom 1000
OK
127.0.0.1:6379> set mic 1000
OK
127.0.0.1:6379> multi
OK
127.0.0.1:6379> decrby tom 100
QUEUED
127.0.0.1:6379> incrby mic 100
QUEUED
127.0.0.1:6379> exec
1) (integer) 900
2) (integer) 1100
127.0.0.1:6379> get tom
"900"
127.0.0.1:6379> get mic
"1100"

通过multi的命令开启事务,multi执行后,客户端可以继续向服务器发送任意多条命令,这些命令不会立即执行,而是被放到一个队列中。当exec命令被调用时,所有队列中的命令才会被执行。

如果没有执行exec命令,所有的命令都不会被执行。

2、事务取消

如果中途不想执行事务了,怎么办?可以使用discard命令清空事务队列,放弃执行。

127.0.0.1:6379> set tom 1000
OK
127.0.0.1:6379> set mic 1000
OK
127.0.0.1:6379> multi
OK
127.0.0.1:6379> decrby tom 100
QUEUED
127.0.0.1:6379> discard
OK
127.0.0.1:6379> get tom
"1000"

3、watch命令

为了防止事务过程中某个key的值被其他客户端请求修改,带来非预期的结果,在Redis中海提供了一个watch命令。

也就是说,多个客户端更新变量的时候,会跟原值做比较,只有它没有被其他线程修改的情况下,才更新成新的值。它可以为Redis事务提供CAS乐观锁的行为(Compare and Swap)。

我们可以用watch监视一个或多个key,如果开启事务之后,至少有一个被监视key键在exec执行之前被修改了,那么整个事务都会被取消(key提前过期除外)。可以用unwatch取消。

client-1client-2
127.0.0.1:6379> set balance 1000 OK 127.0.0.1:6379> watch balance OK 127.0.0.1:6379> multi OK 127.0.0.1:6379> incrby balance 100 QUEUED
\127.0.0.1:6379> decrby balance 100 (integer) 900
127.0.0.1:6379> exec (nil) 127.0.0.1:6379> get balance “900”

三、事务异常

我们都知道,mysql事务执行时发生异常之后,会自动回滚。
Redis事务在执行的过程中发生了异常,会回滚吗?

事务执行遇到的问题一般分为两种,一种是在执行exec之前发生错误,一种是在执行exec之后发生错误。

1、在执行exec之前发生错误

比如:入队的命令存在语法错误,包括参数数量、参数名等等(编译器错误)。

127.0.0.1:6379> multi
OK
127.0.0.1:6379> set zhangsan 123
QUEUED
127.0.0.1:6379> set lisi 321
QUEUED
127.0.0.1:6379> hset wangwu 333
(error) ERR wrong number of arguments for 'hset' command
127.0.0.1:6379> exec
(error) EXECABORT Transaction discarded because of previous errors.

我们会发现,hset我们的语法有问题,事务会被拒绝执行,也就是队列中所有的命令都不会得到执行。

2、在执行exec之后发生错误

比如:对String使用了hash命令,参数个数正确,但是数据类型错误,这是一种运行时错误。

127.0.0.1:6379> multi
OK
127.0.0.1:6379> set a 1
QUEUED
127.0.0.1:6379> hset a aa bb
QUEUED
127.0.0.1:6379> exec
1) OK
2) (error) WRONGTYPE Operation against a key holding the wrong kind of value
127.0.0.1:6379> get a
"1"

我们发现,第一条命令是执行成功的,也就是在发生异常的情况下,之后错误的命令没有被执行,正确的命令还是执行成功,并没有被回滚。

这显然不符合我们对原子性的定义。也就是没办法用Redis的这种事务机制来实现原子性,保证数据一致。

3、为什么不回滚

官方的解释:

  • Redis命令只会因为错误的语法而失败,也就是说,从实用性的角度来讲,失败的命令是由代码错误造成的,而这些错误应该在开发的过程中被发现,而不应该出现在生产环境中(程序员的锅)。
  • 因为不需要对回滚进行支持,所以Redis的内部可以保持简单快速。需要知道的是:回滚不能解决代码的问题(程序员的锅必须程序员来背,不应当交由Redis开发者来规避)。