You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

246 lines
16 KiB
Markdown

This file contains ambiguous Unicode characters!

This file contains ambiguous Unicode characters that may be confused with others in your current locale. If your use case is intentional and legitimate, you can safely ignore this warning. Use the Escape button to highlight these characters.

# 31 | 事务机制Redis能实现ACID属性吗
你好,我是蒋德钧。
事务是数据库的一个重要功能。所谓的事务就是指对数据进行读写的一系列操作。事务在执行时会提供专门的属性保证包括原子性Atomicity、一致性Consistency、隔离性Isolation和持久性Durability也就是ACID属性。这些属性既包括了对事务执行结果的要求也有对数据库在事务执行前后的数据状态变化的要求。
那么Redis可以完全保证ACID属性吗毕竟如果有些属性在一些场景下不能保证的话很可能会导致数据出错所以我们必须要掌握Redis对这些属性的支持情况并且提前准备应对策略。
接下来我们就先了解ACID属性对事务执行的具体要求有了这个知识基础后我们才能准确地判断Redis的事务机制能否保证ACID属性。
## 事务ACID属性的要求
首先来看原子性。原子性的要求很明确,就是一个事务中的多个操作必须都完成,或者都不完成。业务应用使用事务时,原子性也是最被看重的一个属性。
我给你举个例子。假如用户在一个订单中购买了两个商品A和B那么数据库就需要把这两个商品的库存都进行扣减。如果只扣减了一个商品的库存那么这个订单完成后另一个商品的库存肯定就错了。
第二个属性是一致性。这个很容易理解,就是指数据库中的数据在事务执行前后是一致的。
第三个属性是隔离性。它要求数据库在执行一个事务时,其它操作无法存取到正在执行事务访问的数据。
我还是借助用户下单的例子给你解释下。假设商品A和B的现有库存分别是5和10用户X对A、B下单的数量分别是3、6。如果事务不具备隔离性在用户X下单事务执行的过程中用户Y一下子也购买了5件B这和X购买的6件B累加后就超过B的总库存值了这就不符合业务要求了。
最后一个属性是持久性。数据库执行事务后,数据的修改要被持久化保存下来。当数据库重启后,数据的值需要是被修改后的值。
了解了ACID属性的具体要求后我们再来看下Redis是如何实现事务机制的。
## Redis如何实现事务
事务的执行过程包含三个步骤Redis提供了MULTI、EXEC两个命令来完成这三个步骤。下面我们来分析下。
第一步客户端要使用一个命令显式地表示一个事务的开启。在Redis中这个命令就是MULTI。
第二步客户端把事务中本身要执行的具体操作例如增删改数据发送给服务器端。这些操作就是Redis本身提供的数据读写命令例如GET、SET等。不过这些命令虽然被客户端发送到了服务器端但Redis实例只是把这些命令暂存到一个命令队列中并不会立即执行。
第三步客户端向服务器端发送提交事务的命令让数据库实际执行第二步中发送的具体操作。Redis提供的**EXEC命令**就是执行事务提交的。当服务器端收到EXEC命令后才会实际执行命令队列中的所有命令。
下面的代码就显示了使用MULTI和EXEC执行一个事务的过程你可以看下。
```
#开启事务
127.0.0.1:6379> MULTI
OK
#将a:stock减1
127.0.0.1:6379> DECR a:stock
QUEUED
#将b:stock减1
127.0.0.1:6379> DECR b:stock
QUEUED
#实际执行事务
127.0.0.1:6379> EXEC
1) (integer) 4
2) (integer) 9
```
我们假设a:stock、b:stock两个键的初始值是5和10。在MULTI命令后执行的两个DECR命令是把a:stock、b:stock两个键的值分别减1它们执行后的返回结果都是QUEUED这就表示这些操作都被暂存到了命令队列还没有实际执行。等到执行了EXEC命令后可以看到返回了4、9这就表明两个DECR命令已经成功地执行了。
好了通过使用MULTI和EXEC命令我们可以实现多个操作的共同执行但是这符合事务要求的ACID属性吗接下来我们就来具体分析下。
## Redis的事务机制能保证哪些属性
原子性是事务操作最重要的一个属性所以我们先来分析下Redis事务机制能否保证原子性。
### 原子性
如果事务正常执行没有发生任何错误那么MULTI和EXEC配合使用就可以保证多个操作都完成。但是如果事务执行发生错误了原子性还能保证吗我们需要分三种情况来看。
第一种情况是,**在执行EXEC命令前客户端发送的操作命令本身就有错误**比如语法错误使用了不存在的命令在命令入队时就被Redis实例判断出来了。
对于这种情况在命令入队时Redis就会报错并且记录下这个错误。此时我们还能继续提交命令操作。等到执行了EXEC命令之后Redis就会拒绝执行所有提交的命令操作返回事务失败的结果。这样一来事务中的所有命令都不会再被执行了保证了原子性。
我们来看一个因为事务操作入队时发生错误,而导致事务失败的小例子。
```
#开启事务
127.0.0.1:6379> MULTI
OK
#发送事务中的第一个操作但是Redis不支持该命令返回报错信息
127.0.0.1:6379> PUT a:stock 5
(error) ERR unknown command `PUT`, with args beginning with: `a:stock`, `5`,
#发送事务中的第二个操作这个操作是正确的命令Redis把该命令入队
127.0.0.1:6379> DECR b:stock
QUEUED
#实际执行事务但是之前命令有错误所以Redis拒绝执行
127.0.0.1:6379> EXEC
(error) EXECABORT Transaction discarded because of previous errors.
```
在这个例子中事务里包含了一个Redis本身就不支持的PUT命令所以在PUT命令入队时Redis就报错了。虽然事务里还有一个正确的DECR命令但是在最后执行EXEC命令后整个事务被放弃执行了。
我们再来看第二种情况。
和第一种情况不同的是,**事务操作入队时命令和操作的数据类型不匹配但Redis实例没有检查出错误**。但是在执行完EXEC命令以后Redis实际执行这些事务操作时就会报错。不过需要注意的是虽然Redis会对错误命令报错但还是会把正确的命令执行完。在这种情况下事务的原子性就无法得到保证了。
举个小例子。事务中的LPOP命令对String类型数据进行操作入队时没有报错但是在EXEC执行时报错了。LPOP命令本身没有执行成功但是事务中的DECR命令却成功执行了。
```
#开启事务
127.0.0.1:6379> MULTI
OK
#发送事务中的第一个操作LPOP命令操作的数据类型不匹配此时并不报错
127.0.0.1:6379> LPOP a:stock
QUEUED
#发送事务中的第二个操作
127.0.0.1:6379> DECR b:stock
QUEUED
#实际执行事务,事务第一个操作执行报错
127.0.0.1:6379> EXEC
1) (error) WRONGTYPE Operation against a key holding the wrong kind of value
2) (integer) 8
```
看到这里你可能有个疑问传统数据库例如MySQL在执行事务时会提供回滚机制当事务执行发生错误时事务中的所有操作都会撤销已经修改的数据也会被恢复到事务执行前的状态那么在刚才的例子中如果命令实际执行时报错了是不是可以用回滚机制恢复原来的数据呢
其实Redis中并没有提供回滚机制。虽然Redis提供了DISCARD命令但是这个命令只能用来主动放弃事务执行把暂存的命令队列清空起不到回滚的效果。
DISCARD命令具体怎么用呢我们来看下下面的代码。
```
#读取a:stock的值4
127.0.0.1:6379> GET a:stock
"4"
#开启事务
127.0.0.1:6379> MULTI
OK
#发送事务的第一个操作对a:stock减1
127.0.0.1:6379> DECR a:stock
QUEUED
#执行DISCARD命令主动放弃事务
127.0.0.1:6379> DISCARD
OK
#再次读取a:stock的值值没有被修改
127.0.0.1:6379> GET a:stock
"4"
```
这个例子中a:stock键的值一开始为4然后我们执行一个事务想对a:stock的值减1。但是在事务的最后我们执行的是DISCARD命令所以事务就被放弃了。我们再次查看a:stock的值会发现仍然为4。
最后,我们再来看下第三种情况:**在执行事务的EXEC命令时Redis实例发生了故障导致事务执行失败**。
在这种情况下如果Redis开启了AOF日志那么只会有部分的事务操作被记录到AOF日志中。我们需要使用redis-check-aof工具检查AOF日志文件这个工具可以把未完成的事务操作从AOF文件中去除。这样一来我们使用AOF恢复实例后事务操作不会再被执行从而保证了原子性。
当然如果AOF日志并没有开启那么实例重启后数据也都没法恢复了此时也就谈不上原子性了。
好了到这里你了解了Redis对事务原子性属性的保证情况我们来简单小结下
* 命令入队时就报错,会放弃事务执行,保证原子性;
* 命令入队时没报错,实际执行时报错,不保证原子性;
* EXEC命令执行时实例故障如果开启了AOF日志可以保证原子性。
接下来,我们再来学习下一致性属性的保证情况。
### 一致性
事务的一致性保证会受到错误命令、实例故障的影响。所以,我们按照命令出错和实例故障的发生时机,分成三种情况来看。
**情况一:命令入队时就报错**
在这种情况下,事务本身就会被放弃执行,所以可以保证数据库的一致性。
**情况二:命令入队时没报错,实际执行时报错**
在这种情况下,有错误的命令不会被执行,正确的命令可以正常执行,也不会改变数据库的一致性。
**情况三EXEC命令执行时实例发生故障**
在这种情况下实例故障后会进行重启这就和数据恢复的方式有关了我们要根据实例是否开启了RDB或AOF来分情况讨论下。
如果我们没有开启RDB或AOF那么实例故障重启后数据都没有了数据库是一致的。
如果我们使用了RDB快照因为RDB快照不会在事务执行时执行所以事务命令操作的结果不会被保存到RDB快照中使用RDB快照进行恢复时数据库里的数据也是一致的。
如果我们使用了AOF日志而事务操作还没有被记录到AOF日志时实例就发生了故障那么使用AOF日志恢复的数据库数据是一致的。如果只有部分操作被记录到了AOF日志我们可以使用redis-check-aof清除事务中已经完成的操作数据库恢复后也是一致的。
所以总结来说在命令执行错误或Redis发生故障的情况下Redis事务机制对一致性属性是有保证的。接下来我们再继续分析下隔离性。
### 隔离性
事务的隔离性保证会受到和事务一起执行的并发操作的影响。而事务执行又可以分成命令入队EXEC命令执行前和命令实际执行EXEC命令执行后两个阶段所以我们就针对这两个阶段分成两种情况来分析
1. 并发操作在EXEC命令前执行此时隔离性的保证要使用WATCH机制来实现否则隔离性无法保证
2. 并发操作在EXEC命令后执行此时隔离性可以保证。
我们先来看第一种情况。一个事务的EXEC命令还没有执行时事务的命令操作是暂存在命令队列中的。此时如果有其它的并发操作我们就需要看事务是否使用了WATCH机制。
WATCH机制的作用是在事务执行前监控一个或多个键的值变化情况当事务调用EXEC命令执行时WATCH机制会先检查监控的键是否被其它客户端修改了。如果修改了就放弃事务执行避免事务的隔离性被破坏。然后客户端可以再次执行事务此时如果没有并发修改事务数据的操作了事务就能正常执行隔离性也得到了保证。
WATCH机制的具体实现是由WATCH命令实现的我给你举个例子你可以看下下面的图进一步理解下WATCH命令的使用。
![](https://static001.geekbang.org/resource/image/4f/73/4f8589410f77df16311dd29131676373.jpg)
我来给你具体解释下图中的内容。
在t1时客户端X向实例发送了WATCH命令。实例收到WATCH命令后开始监测a:stock的值的变化情况。
紧接着在t2时客户端X把MULTI命令和DECR命令发送给实例实例把DECR命令暂存入命令队列。
在t3时客户端Y也给实例发送了一个DECR命令要修改a:stock的值实例收到命令后就直接执行了。
等到t4时实例收到客户端X发送的EXEC命令但是实例的WATCH机制发现a:stock已经被修改了就会放弃事务执行。这样一来事务的隔离性就可以得到保证了。
当然如果没有使用WATCH机制在EXEC命令前执行的并发操作是会对数据进行读写的。而且在执行EXEC命令的时候事务要操作的数据已经改变了在这种情况下Redis并没有做到让事务对其它操作隔离隔离性也就没有得到保障。下面这张图显示了没有WATCH机制时的情况你可以看下。
![](https://static001.geekbang.org/resource/image/8c/57/8ca37debfff91282b9c62a25fd7e9a57.jpg)
在t2时刻客户端X发送的EXEC命令还没有执行但是客户端Y的DECR命令就执行了此时a:stock的值会被修改这就无法保证X发起的事务的隔离性了。
刚刚说的是并发操作在EXEC命令前执行的情况下面我再来说一说第二种情况**并发操作在EXEC命令之后被服务器端接收并执行**。
因为Redis是用单线程执行命令而且EXEC命令执行后Redis会保证先把命令队列中的所有命令执行完。所以在这种情况下并发操作不会破坏事务的隔离性如下图所示
![](https://static001.geekbang.org/resource/image/11/ae/11a1eff930920a0b423a6e46c23f44ae.jpg)
最后我们来分析一下Redis事务的持久性属性保证情况。
### 持久性
因为Redis是内存数据库所以数据是否持久化保存完全取决于Redis的持久化配置模式。
如果Redis没有使用RDB或AOF那么事务的持久化属性肯定得不到保证。如果Redis使用了RDB模式那么在一个事务执行后而下一次的RDB快照还未执行前如果发生了实例宕机这种情况下事务修改的数据也是不能保证持久化的。
如果Redis采用了AOF模式因为AOF模式的三种配置选项no、everysec和always都会存在数据丢失的情况所以事务的持久性属性也还是得不到保证。
所以不管Redis采用什么持久化模式事务的持久性属性是得不到保证的。
## 小结
在这节课上我们学习了Redis中的事务实现。Redis通过MULTI、EXEC、DISCARD和WATCH四个命令来支持事务机制这4个命令的作用我总结在下面的表中你可以再看下。
![](https://static001.geekbang.org/resource/image/95/50/9571308df0620214d7ccb2f2cc73a250.jpg)
事务的ACID属性是我们使用事务进行正确操作的基本要求。通过这节课的分析我们了解到了Redis的事务机制可以保证一致性和隔离性但是无法保证持久性。不过因为Redis本身是内存数据库持久性并不是一个必须的属性我们更加关注的还是原子性、一致性和隔离性这三个属性。
原子性的情况比较复杂,只有当事务中使用的命令语法有误时,原子性得不到保证,在其它情况下,事务都可以原子性执行。
所以,我给你一个小建议:**严格按照Redis的命令规范进行程序开发并且通过code review确保命令的正确性**。这样一来Redis的事务机制就能被应用在实践中保证多操作的正确执行。
## 每课一问
按照惯例我给你提个小问题在执行事务时如果Redis实例发生故障而Redis使用了RDB机制那么事务的原子性还能得到保证吗
欢迎在留言区写下你的思考和答案,我们一起交流讨论。如果你觉得今天的内容对你有所帮助,也欢迎你分享给你的朋友或同事。我们下节课见。