数据库
首页 > 数据库> > 【Redis】事务和锁

【Redis】事务和锁

作者:互联网

纸上得来终觉浅,绝知此事要躬行。

什么是事务

事务可以一次执行多个命令,本质是一组命令的集合, 并且带有以下两个重要的保证:

为什么要事务

首先我们先来看一个案例:

session-1 session-2
127.0.0.1:6379> set money 1000
OK
127.0.0.1:6379> set money 1
OK
127.0.0.1:6379> get money
"1"
127.0.0.1:6379> get money
"1"

上面相当于两个客户端同时进行,session-1为客户端1,session-2为客户端2。客户端1先set money 1000,紧接着客户端2执行set money 1,当客户端1通过get money查看的时候最终的值变成了1,也就是说客户端1在执行的过程中被客户端2插队了,有没有办法保证一个客户端在执行过程中整个操作是一个整体,有,就是下面需要了解的事务以及事务相关的操作命令。

事务的操作

MULTI

127.0.0.1:6379> MULTI
OK
127.0.0.1:6379> set age 22
QUEUED
127.0.0.1:6379> INCR age
QUEUED
127.0.0.1:6379> get age
QUEUED
127.0.0.1:6379> 

EXEC

127.0.0.1:6379> MULTI
OK
127.0.0.1:6379> set age 22
QUEUED
127.0.0.1:6379> INCR age
QUEUED
127.0.0.1:6379> get age
QUEUED
127.0.0.1:6379> EXEC
1) OK
2) (integer) 23
3) "23"
127.0.0.1:6379>

DISCARD

127.0.0.1:6379> MULTI
OK
127.0.0.1:6379> set age 22
QUEUED
127.0.0.1:6379> INCR age
QUEUED
127.0.0.1:6379> get age
QUEUED
127.0.0.1:6379> DISCARD
OK
127.0.0.1:6379> EXEC
(error) ERR EXEC without MULTI
127.0.0.1:6379> 

事务工作流程

事务工作流程

事务中的错误

使用事务时可能会遇上以下两种错误:

127.0.0.1:6379> set name ydongy
QUEUED
127.0.0.1:6379> aaa name ydongy2
(error) ERR unknown command `aaa`, with args beginning with: `name`, `ydongy2`, 
127.0.0.1:6379> EXEC
(error) EXECABORT Transaction discarded because of previous errors.
127.0.0.1:6379> 

我们发现,当事务中出现语法错误,最后exec提示事务不存在,也就是说在事务中执行语法错误,整体事务中所有命令均不会执行。包括那些语法正确的命令

127.0.0.1:6379> MULTI
OK
127.0.0.1:6379> set name ydongy
QUEUED
127.0.0.1:6379> get name
QUEUED
127.0.0.1:6379> set age 22
QUEUED
127.0.0.1:6379> get age
QUEUED
127.0.0.1:6379> lpush name 1 2 3
QUEUED
127.0.0.1:6379> get name
QUEUED
127.0.0.1:6379> EXEC
1) OK
2) "ydongy"
3) OK
4) "22"
5) (error) WRONGTYPE Operation against a key holding the wrong kind of value
6) "ydongy"
127.0.0.1:6379> 

上面的案例有一步错误操作,就是把name当成列表进行追加数据,语法本身没有错,但是在最后执行的时候那一条指令并没有执行,但是整个事务中正确的指令都执行了。需要注意已经执行完毕的命令对应的数据不会自动回滚,即 Redis 不支持回滚,需要程序员自己在代码中实现回滚。

业务场景:

业务场景

为了解决这种并发操作带来的问题,Redis 的 WATCH 命令可以为事务提供 check-and-set (CAS)行为,即:乐观锁。被 WATCH 的键会被监视,并会发觉这些键是否被改动过了。 如果有至少一个被监视的键在 EXEC 执行之前被修改了, 那么整个事务都会被取消。

session-1 session-2
127.0.0.1:6379> set age 22
OK
127.0.0.1:6379> get age
"22"
127.0.0.1:6379> WATCH age
OK
127.0.0.1:6379> MULTI
OK
127.0.0.1:6379> INCR age
QUEUED
127.0.0.1:6379> DECR age
(integer) 21
127.0.0.1:6379> EXEC
(nil)
127.0.0.1:6379> get age
"21"
127.0.0.1:6379> get age
"21"

通过EXEC 返回nil来表示事务已经失败。当 EXEC 被调用时,不管事务是否成功执行,对所有键的监视都会被取消。另外,当客户端断开连接时,该客户端对键的监视也会被取消。如果需要WATCH 对多个键进行监视,可以使用无参数的 UNWATCH 命令手动取消对所有键的监视。

标签:事务,127.0,0.1,age,Redis,6379,QUEUED
来源: https://www.cnblogs.com/ydongy/p/13229574.html