前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Redis 事务(8)

Redis 事务(8)

作者头像
兜兜毛毛
发布2020-03-19 10:38:43
5110
发布2020-03-19 10:38:43
举报
文章被收录于专栏:兜兜毛毛

为什么要用事务

Redis的单个命令是原子性的(比如get set mget mset),如果涉及到多个命令的时候,需要把多个命令作为一个不可分割的处理序列,就需要用到事务。

例如我们之前说的用setnx实现分布式锁,我们先set,然后设置对key设置expire,防止del发生异常的时候锁不会被释放,业务处理完了以后再del,这三个动作我们就希望它们作为一组命令执行。

Redis的事务有两个特点:

  1. 按进入队列的顺序执行。
  2. 不会受到其他客户端的请求的影响。
Redis的事务涉及到四个命令:

命令

说明

multi

开启事务

exec

执行事务

discard

取消事务

watch

监视

事务用法

案例:张三(zhangsan)和李四(lisi)各有100元,张三需向李四转账50元。张三账户余额减少50元,李四的账户余额增加50元。

代码语言:javascript
复制
127.0.0.1:6379> set zhangsan 100
OK
127.0.0.1:6379> set lisi 100
OK
127.0.0.1:6379> multi
OK
127.0.0.1:6379> decrby zhangsan 50
QUEUED
127.0.0.1:6379> incrby lisi 50
QUEUED
127.0.0.1:6379> exec
1) (integer) 50
2) (integer) 150
127.0.0.1:6379> get zhangsan
"50"
127.0.0.1:6379> get lisi
"150"

通过multi的命令开启事务。事务不能嵌套,多个multi命令效果一样。

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

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

如果中途不想执行事务了,怎么办?

可以调用discard可以清空事务队列,放弃执行。

代码语言:javascript
复制
127.0.0.1:6379> multi
OK
127.0.0.1:6379> set s1 1
QUEUED
127.0.0.1:6379> set s2 2
QUEUED
127.0.0.1:6379> set s3 3
QUEUED
127.0.0.1:6379> discard
OK
127.0.0.1:6379> get s1
(error) WRONGTYPE Operation against a key holding the wrong kind of value

开通两个客户端

代码语言:javascript
复制
####------>>>>>第一个客户端执行命令
127.0.0.1:6379> set zhangsan 1000
OK
127.0.0.1:6379> get zhangsan
"1000"
127.0.0.1:6379> watch zhangsan
OK
127.0.0.1:6379> multi
OK
127.0.0.1:6379> incrby zhangsan 100
QUEUED

####------>>>>>第二个客户端执行命令
127.0.0.1:6379> decrby zhangsan 100
(integer) 900

####------>>>>>第一个客户端执行命令
127.0.0.1:6379> exec
(nil)
127.0.0.1:6379> get zhangsan
"900"
事务可能遇到的问题

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

在执行exec之前发生错误

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

代码语言:javascript
复制
127.0.0.1:6379> multi
OK
127.0.0.1:6379> set list 111
QUEUED
127.0.0.1:6379> hset list 222
(error) ERR wrong number of arguments for 'hset' command
127.0.0.1:6379> exec
(error) EXECABORT Transaction discarded because of previous errors.

在这种情况下事务会被拒绝执行,也就是队列中所有的命令都不会得到执行。

在执行exec之后发生错误

比如,类型错误,比如对String使用了Hash的命令,这是一种运行时错误。

代码语言:javascript
复制
127.0.0.1:6379> flushall
OK
127.0.0.1:6379> multi
OK
127.0.0.1:6379> set s1 1
QUEUED
127.0.0.1:6379> hset s1 a b
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 s1
"1"

最后我们发现setk11的命令是成功的,也就是在这种发生了运行时异常的情况下,只有错误的命令没有被执行,但是其他命令没有受到影响。

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

为什么在一个事务中存在错误,Redis不回滚?

这种方式也有其合理之处:只有当被调用的Redis命令有语法错误时,这条命令才会执行失败(在将这个命令放入事务队列期间,Redis能够发现此类问题),或者对某个键执行不符合其数据类型的操作:实际上,这就意味着只有程序错误才会导致Redis命令执行失败,这种错误很有可能在程序开发期间发现,一般很少在生产环境发现。

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • Redis的事务涉及到四个命令:
  • 事务用法
  • 事务可能遇到的问题
    • 在执行exec之前发生错误
      • 在执行exec之后发生错误
        • 为什么在一个事务中存在错误,Redis不回滚?
        相关产品与服务
        云数据库 Redis
        腾讯云数据库 Redis(TencentDB for Redis)是腾讯云打造的兼容 Redis 协议的缓存和存储服务。丰富的数据结构能帮助您完成不同类型的业务场景开发。支持主从热备,提供自动容灾切换、数据备份、故障迁移、实例监控、在线扩容、数据回档等全套的数据库服务。
        领券
        问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档