redisbook(redis什么时候要用事务)
本文目录
redis什么时候要用事务
Redis事务使用总结:
Redis的事务机制允许同时执行多条指令,它是原子性操作,事务中的命令要么全部执行,要么全部不执行,另外,事务中的所有指令都会被序列化,而且其开始执行过程中,不回被即时过来的指令所打断,其需要经历三个过程,分别为开始事务、命令入队以及执行事务。
· 相关命令
· 如何使用
· 脚本事务
· 遇到问题
· 例子演示
一、相关命令
1、MULTI
该命令用来开启事务,它总是返回ok结果,当其执行之后,客户端可以继续发送任意条数量的指令,这些指令不会立即被执行,而是被放到了队列中,直到EXEC被调用之后,所有命令才会被序列化执行。
2、EXEC
该命令负责触发并执行队列中所有的命令。
NOTE:
如果MULTI开启之后,因为某些原因没有成功执行EXEC,那么事务中所有的命令都不会被执行的。
3、DISCARD
该命令用来刷新事务中所有排队等待执行的指令,它总是返回ok结果,并且将服务连接状态恢复到正常。如果已经使用WATCH,那么其会将释放所有被WATCH的key。
4、WATCH
标记所有指定的key被监控起来,使其在事务中有条件的执行(乐观锁)。
NOTE:
A、WATCH使得EXEC命令需要有条件的执行,也就是事务只能在所有被监视的键没有被修改的前提下才能执行。另外,在EXEC被执行之后,所有的WATCH都会被取消。
B、UNWATCH手动取消对所有键的WATCH,如果执行了EXEC或者DISCARD,则不需要手动执行UNWATCH命令。
二、如何使用
Redis原生使用(Redis-cli):
127.0.0.1:6379》 multi // 事务开始的动作标志下面即为入队
OK
127.0.0.1:6379》 set book-name "Thinking in Java"
QUEUED
127.0.0.1:6379》 get book-name
QUEUED
127.0.0.1:6379》 sadd tag "java" "Programming""Thinking"
QUEUED
127.0.0.1:6379》 smembers tag
QUEUED
127.0.0.1:6379》 exec // 执行事务
1) OK
2) "Thinking in Java"
3) (integer) 3
4) 1) "Thinking"
2) "Programming"
3) "java"
127.0.0.1:6379》 discard // 事务已执行完毕 已经自动取消
(error) ERR DISCARD without MULTI
127.0.0.1:6379》 multi
OK
127.0.0.1:6379》 set book-name "Patterns in Java"
QUEUED
127.0.0.1:6379》 get book-name
QUEUED
127.0.0.1:6379》 sadd tag "Java" "Thinking""Programming"
QUEUED
127.0.0.1:6379》 smembers tag
QUEUED
127.0.0.1:6379》 discard // 事务未执行 可以刷新队列指令状态 取消执行
OK
127.0.0.1:6379》 exec // 事务已经被取消不能再执行
(error) ERR EXEC without MULTI
三、脚本事务
Redis 2.6开始支持了脚本,而该脚本本身就是一种事务机制,所以任何在事务里可以完成的事,在脚本里面也能完成,并且使用脚本更简单些,并且速度也更快。不过因为事务提供了一种即使不使用脚本,也可以避免竞争条件的方法,并且事务本身的实现并不复杂,所以现在的使用也比较多,但不排除日后可能被替代或是占据主要地位的可能。
NOTE:
Redis为什么引入两种处理事务的方式?脚本功能是 Redis 2.6 才引入的,而事务功能则在更早之前就存在,所以 Redis 才会同时存在两种处理事务的方法。另外,事务脚本会在后续文章中总结介绍。
四、遇到问题
1、乐观锁实现
举个例子,假设我们需要原子性为某个键加1操作(假设INCR不存在),那么应该是这样的执行语句:
SET mykey 1
val = GET mykey
val = val + 1
SET mykey ${val}
单个客户端访问操作没有任何问题,如果是多个客户端同时访问mykey,就会产生资源共享访问问题,比如:现在有个两个客户端访问同一个键mykey,那么mykey的可能是2,但是我们期望的值应该是3才对,这个类似于高并发下的sync锁机制,所以我们需要使用WATCH来监控被共享的键mykey,如下:
WATCH mykey(可监控多个键)
val = GET mykey
val = val + 1
MULTI
SET mykey ${val}
EXEC
NOTE:
虽然大多情况下,多个客户端访问操作同一个键的情况很少或没有,但是不能排除这个特殊情况,所以建议在有可能产生键共享的指令中使用WATCH在EXEC执行前对其监管。
2、Redis不支持回滚(Roll Back)
Redis的事务不支持回滚,这点不同于关系数据库中的事务,所以它的内部保持了简单且快速的特点。另外,Redis不支持回滚是这样考虑的:Redis事务中命令之所以会失败,是由于错误的编程所造成,通过事务回滚是不能回避这个根本问题。
NOTE:
Redis事务中命令执行失败,仍会继续执行后面的执行,在没有特殊干预前提下,直到执行完队列中所有指令为止。
3、使用事务可能遇到的问题
A、事务在执行 EXEC 之前,入队的命令可能会出错,举个例子:命令可能会产生语法错误(参数数量错误,参数名错误等),或者其他更严重的错误,比如内存不足(如果服务器使用maxmemory 设置了最大内存限制的话)。
B、事务在执行 EXEC 之前,举个例子:事务中的命令可能处理了错误类型的键,比如将列表命令用在了字符串键上面等。
对于发生在 EXEC 执行之前的错误,客户端以前的做法是检查命令入队所得的返回值:如果命令入队时返回QUEUED ,那么入队成功;否则,就是入队失败。如果有命令在入队时失败,那么大部分客户端都会停止并取消这个事务。
从 Redis 2.6.5 开始,服务器会对命令入队失败的情况进行记录,并在客户端调用 EXEC 命令时,拒绝执行并自动放弃这个事务。
在 Redis 2.6.5 以前, Redis 只执行事务中那些入队成功的命令,而忽略那些入队失败的命令。而新的处理方式则使得在管道技术中包含事务变得简单,因为发送事务和读取事务的回复都只需要和服务器进行一次通讯即可。
至于那些在 EXEC 命令执行之后所产生的错误,并没有对它们进行特别处理: 即使事务中有某个/某些命令在执行时产生了错误, 事务中的其他命令仍然会继续执行。
五、例子演示
《?php
$redis = new \Redis();
$redis-》connect(’127.0.0.1’,6379);
$result = array();
// 开启事务
$redis-》multi();
// 添加指令到队列
$redis-》set(’book-name’,’Thinking in PHP!’);
$redis-》sAdd(’tags’,’PHP’,’Programming’,’Thinking’);
$bookname = $redis-》get(’book-name’);
$tags = $redis-》sMembers(’tags’);
// 执行事务
$redis-》exec();
// 显示结果
echo ’书名:’.$bookname.’ 标签:’.$tags;
?》
结果:
redis有哪些存储模式
Redis支持多种数据结构和存储模式,其中包括:
字符串(String):字符串类型是Redis最基本的数据类型,它可以包含任何数据,比如文本、整数或二进制数据等。
哈希(Hash):哈希类型存储的是键值对集合,这些键值对可以是字符串类型的,也可以是数字类型的。
列表(List):列表类型是一个有序的字符串列表,可以添加、删除和插入元素。
集合(Set):集合类型存储的是一组唯一的无序元素,支持添加、删除和查询操作。
有序集合(Sorted Set):有序集合类型存储的是一组有序的元素,每个元素都有一个分数(score),可以根据分数进行排序。
RDB持久化模式:在指定时间间隔内将内存中的数据保存到磁盘中。
AOF持久化模式:将所有对Redis数据库的写操作记录下来,可以通过回放这些日志文件来恢复数据库。
混合持久化模式:同时使用RDB和AOF两种持久化模式,以保证数据的可靠性和恢复速度。
此外,Redis还支持多种不同的持久化模式,包括:
Macbook Pro 终端安装redis,启动redis-server之后,为啥不能用别的命令了
清空数据库:打开redis-client.exe,输入命令flushdb redis自带持久化,因此是不会因为掉电而丢失所有内容的。默认情况下,redis每隔一定时间间隔会对数据库做一次内存快照,记录数据库此刻的内容;启动服务器时会自动加载内存快照。 你可以打开...
mac 启动 redis
1、先找到redis的安装位置 默认安装的位置是 /usr/local/bin 2、找到带有 redis-server 的目录里,直接 输入 redis-server 即可
更多文章:
![surfacepro6与pro7的区别(买Surface Pro6还是Surface Pro7好呀纠结中)](/static/images/nopic/26.jpg)
surfacepro6与pro7的区别(买Surface Pro6还是Surface Pro7好呀纠结中)
2024年2月29日 05:50
![苹果ipad pro 2020(ipadpro2020屏幕护眼吗)](/static/images/nopic/22.jpg)
苹果ipad pro 2020(ipadpro2020屏幕护眼吗)
2024年6月8日 15:50
![dacom耳机怎么样(Dacom Athlete+运动蓝牙耳机可以用多长时间)](/static/images/nopic/22.jpg)
dacom耳机怎么样(Dacom Athlete+运动蓝牙耳机可以用多长时间)
2024年5月29日 21:00
![htcg13是什么上市的(HTC G13,IMEI:359465044374553是哪里产哪里上市的机子)](/static/images/nopic/8.jpg)
htcg13是什么上市的(HTC G13,IMEI:359465044374553是哪里产哪里上市的机子)
2024年4月3日 05:40
![iqoo发布会宋大腿(2799 元起!iQOO Noe6 正式发布)](/static/images/nopic/20.jpg)
iqoo发布会宋大腿(2799 元起!iQOO Noe6 正式发布)
2024年5月30日 01:50
![iphone13屏幕比例是多少(iphone13和13pro尺寸一样吗-iphone13和13pro一样大吗)](/static/images/nopic/19.jpg)
iphone13屏幕比例是多少(iphone13和13pro尺寸一样吗-iphone13和13pro一样大吗)
2024年3月3日 06:20
![诺基亚经典推拉手机(适合女生用的诺基亚手机有什么好看的推拉式手机吗价格在1500之内)](/static/images/nopic/3.jpg)
诺基亚经典推拉手机(适合女生用的诺基亚手机有什么好看的推拉式手机吗价格在1500之内)
2024年3月10日 19:10
![ipad2018能卖多少钱(二手的ipad2018能卖多少128g)](/static/images/nopic/8.jpg)
ipad2018能卖多少钱(二手的ipad2018能卖多少128g)
2024年4月26日 11:10
![三星a90是什么屏幕(三星A90和A51哪个支持120hz)](/static/images/nopic/24.jpg)
三星a90是什么屏幕(三星A90和A51哪个支持120hz)
2024年2月18日 00:50