永发信息网

redis什么时候要用事务

答案:1  悬赏:0  手机版
解决时间 2021-03-26 06:57
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 命令执行之后所产生的错误,并没有对它们进行特别处理: 即使事务中有某个/某些命令在执行时产生了错误, 事务中的其他命令仍然会继续执行。
五、例子演示
$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;
?>
结果:
 
我要举报
如以上问答信息为低俗、色情、不良、暴力、侵权、涉及违法等信息,可以点下面链接进行举报!
大家都在看
建筑基础的埋深是怎么计算的~
编写一个程序,输入一个整数,将该数各位上为
Wii最高支持多大SD卡.也支持SDHC卡吗?
韩文:夸奖别人厉害应该怎么说
百香果可从和红萝卜一起榨汁喝吗
歌词:现在我要惩罚你听这首歌一千遍 求歌名啊
单机穿越cs射击破解版
总觉得尹正好像喜欢韩庚😘只有我一个
【选修4-科学思维方法】我国长江三角洲地区经
怎么对付这种白眼狼?
周末做什么事让假期充实点
蓝领贷昨天晚上申请的怎么还在放款中,求解
56年山下火命和69年大泽土命婚配可以吗
我的华硕10周年主板,内存总出现问题。
2.475乘上三分之一是多少
推荐资讯
工程招标,按标准收费: 若中标金额中已含招
最早运用平行蒙太奇的成功范例是
高危两周了,自测阴,其参考价值有多少
我去年在某宝买了战地1的终极版,也就是革命
金属产品镀镍后 尺寸会有变化吗
380V10平方电线能用多少千瓦
在protel dxp中给PCB电路板覆铜的步骤是什么
《求助》灵宝有几个驾校,哪个比较好
哪一款手机管理软件有自带农历
叛逆和逆叛有什么区别?
和尚娶妻是什么意思
床单被套有没有2.4米乘2.4米的
正方形一边上任一点到这个正方形两条对角线的
阴历怎么看 ?