Redis(三)事务
2022-09-22 22:46:34
目录
[TOC]
事务概念
关系型数据库的事务
比如Mysql是支持事务的,要么同时成功,要么同时失败,一组操作具有原子性!
Redis事务
单条命令是保证原子性的,但一组命令不保证原子性!
Redis事务没有隔离级别的概念!
所有的命令在事务中,并没有直接被执行!只有发起执行命令的时候才会执行!
如何执行事务?
实例
先看一个事务的例子,对于Redis事务有一个实际又具体的理解
1 | # MULTI :开始事务 |
事务经历的三个阶段
一个事务从开始到执行会经历以下三个阶段:
- 开始事务:MULTI
- 命令入队:QUEUED
- 执行事务:EXEC
Redis 事务可以一次执行多个命令, 并且带有以下三个重要的保证:
- 批量操作在发送 EXEC 命令前被放入队列缓存。
- 收到 EXEC 命令后进入事务执行,事务中任意命令执行失败,其余的命令依然被执行。
- 在事务执行过程,其他客户端提交的命令请求不会插入到事务执行命令序列中。
如何结束事务?
不执行EXEC
,而执行DISCARD
1 | 127.0.0.1:6379> multi # 开启事务 |
出错了还会执行吗?
编译型异常:代码有问题,命令写错了,事务中所有的命令都不会执行!
1 | # 查看显示当前库没有数据 |
运行时异常:如果队列中执行时错误(执行时才发现命令执行有问题),那么其他命令还是可以正常执行
1 | # 开启事务 |
再次理解
单个 Redis 命令的执行是原子性的,但 Redis 没有在事务上增加任何维持原子性的机制,所以 Redis 事务的执行并不是原子性的。
事务可以理解为一个打包的批量执行脚本,但批量指令并非原子化的操作,中间某条指令的失败不会导致前面已做指令的回滚,也不会造成后续的指令不做。
EXEC 命令负责触发并执行事务中的所有命令:
- 如果客户端在使用 MULTI 开启了一个事务之后,却因为断线而没有成功执行 EXEC ,那么事务中的所有命令都不会被执行。
- 另一方面,如果客户端成功在开启事务之后执行 EXEC ,那么事务中的所有命令都会被执行。
当使用 AOF 方式做持久化的时候, Redis 会使用单个 write(2) 命令将事务写入到磁盘中。
然而,如果 Redis 服务器因为某些原因被管理员杀死,或者遇上某种硬件故障,那么可能只有部分事务命令会被成功写入到磁盘中。
如果 Redis 在重新启动时发现 AOF 文件出了这样的问题,那么它会退出,并汇报一个错误。
使用
redis-check-aof
程序可以修复这一问题:它会移除 AOF 文件中不完整事务的信息,确保服务器可以顺利启动。