前言

本文主要给大家介绍了关于redis实现加锁的几种方法,分享出来供大家参考学习,下面话不多说了,来一起看看详细的介绍吧。
1. redis加锁分类
redis能用的的加锁命令分表是INCR、SETNX、SET
2. 第一种锁命令INCR
这种加锁的思路是, key 不存在,那么 key 的值会先被初始化为 0 ,然后再执行 INCR 操作进行加一。
然后其它用户在执行 INCR 操作进行加一时,如果返回的数大于 1 ,说明这个锁正在被使用当中。
1、 客户端A请求服务器获取key的值为1表示获取了锁
2、 客户端B也去请求服务器获取key的值为2表示获取锁失败
3、 客户端A执行代码完成,删除锁
4、 客户端B在等待一段时间后在去请求的时候获取key的值为1表示获取锁成功
5、 客户端B执行代码完成,删除锁
$redis->incr($key); $redis->expire($key, $ttl); //设置生成时间为1秒
3. 第二种锁SETNX
这种加锁的思路是,如果 key 不存在,将 key 设置为 value
如果 key 已存在,则 SETNX 不做任何动作
1、 客户端A请求服务器设置key的值,如果设置成功就表示加锁成功
2、 客户端B也去请求服务器设置key的值,如果返回失败,那么就代表加锁失败
3、 客户端A执行代码完成,删除锁
4、 客户端B在等待一段时间后在去请求设置key的值,设置成功
5、 客户端B执行代码完成,删除锁
$redis->setNX($key, $value); $redis->expire($key, $ttl);
4. 第三种锁SET
上面两种方法都有一个问题,会发现,都需要设置 key 过期。那么为什么要设置key过期呢?如果请求执行因为某些原因意外退出了,导致创建了锁但是没有删除锁,那么这个锁将一直存在,以至于以后缓存再也得不到更新。于是乎我们需要给锁加一个过期时间以防不测。
但是借助 Expire 来设置就不是原子性操作了。所以还可以通过事务来确保原子性,但是还是有些问题,所以官方就引用了另外一个,使用 SET 命令本身已经从版本 2.6.12 开始包含了设置过期时间的功能。
1、 客户端A请求服务器设置key的值,如果设置成功就表示加锁成功
2、 客户端B也去请求服务器设置key的值,如果返回失败,那么就代表加锁失败
3、 客户端A执行代码完成,删除锁
4、 客户端B在等待一段时间后在去请求设置key的值,设置成功
5、 客户端B执行代码完成,删除锁
$redis->set($key, $value, array('nx', 'ex' => $ttl)); //ex表示秒
5. 其它问题
虽然上面一步已经满足了我们的需求,但是还是要考虑其它问题?
1、 redis发现锁失败了要怎么办?中断请求还是循环请求?
2、 循环请求的话,如果有一个获取了锁,其它的在去获取锁的时候,是不是容易发生抢锁的可能?
3、 锁提前过期后,客户端A还没执行完,然后客户端B获取到了锁,这时候客户端A执行完了,会不会在删锁的时候把B的锁给删掉?
6. 解决办法
针对问题1:使用循环请求,循环请求去获取锁
针对问题2:针对第二个问题,在循环请求获取锁的时候,加入睡眠功能,等待几毫秒在执行循环
针对问题3:在加锁的时候存入的key是随机的。这样的话,每次在删除key的时候判断下存入的key里的value和自己存的是否一样
do { //针对问题1,使用循环
$timeout = 10;
$roomid = 10001;
$key = 'room_lock';
$value = 'room_'.$roomid; //分配一个随机的值针对问题3
$isLock = Redis::set($key, $value, 'ex', $timeout, 'nx');//ex 秒
if ($isLock) {
if (Redis::get($key) == $value) { //防止提前过期,误删其它请求创建的锁
//执行内部代码
Redis::del($key);
continue;//执行成功删除key并跳出循环
}
} else {
usleep(5000); //睡眠,降低抢锁频率,缓解redis压力,针对问题2
}
} while(!$isLock);
7. 另外一个锁
以上的锁完全满足了需求,但是官方另外还提供了一套加锁的算法,这里以PHP为例
$servers = [
['127.0.0.1', 6379, 0.01],
['127.0.0.1', 6389, 0.01],
['127.0.0.1', 6399, 0.01],
];
$redLock = new RedLock($servers);
//加锁
$lock = $redLock->lock('my_resource_name', 1000);
//删除锁
$redLock->unlock($lock)
上面是官方提供的一个加锁方法,就是和第6的大体方法一样,只不过官方写的更健壮。所以可以直接使用官方提供写好的类方法进行调用。官方提供了各种语言如何实现锁。
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对的支持。
# redis
# 加锁
# redis的分布式锁
# 数据加锁
# 如何用redis setNX命令来加锁
# 浅谈redis加锁常用几种方式
# redis加锁的几种方式汇总
# 客户端
# 在等待
# 值为
# 也去
# 不存在
# 另外一个
# 以防不测
# 都有
# 还没
# 出了
# 还可以
# 说了
# 两种
# 不多
# 会不会
# 作了
# 第二个
# 给大家
# 然后再
相关文章:
如何高效配置香港服务器实现快速建站?
如何快速搭建支持数据库操作的智能建站平台?
建站之星2.7模板:企业网站建设与h5定制设计专题
成都网站制作报价公司,成都工业用气开户费用?
如何选择香港主机高效搭建外贸独立站?
香港服务器租用费用高吗?如何避免常见误区?
如何用AWS免费套餐快速搭建高效网站?
免费ppt制作网站,有没有值得推荐的免费PPT网站?
建站之星如何开启自定义404页面避免用户流失?
建站之星如何一键生成手机站?
魔方云NAT建站如何实现端口转发?
Swift中switch语句区间和元组模式匹配
如何高效配置IIS服务器搭建网站?
C#怎么创建控制台应用 C# Console App项目创建方法
建站之星代理如何获取技术支持?
Android滚轮选择时间控件使用详解
免费制作海报的网站,哪位做平面的朋友告诉我用什么软件做海报比较好?ps还是cd还是ai这几个软件我都会些我是做网页的?
建站一年半SEO优化实战指南:核心词挖掘与长尾流量提升策略
网站制作哪家好,cc、.co、.cm哪个域名更适合做网站?
如何通过NAT技术实现内网高效建站?
如何打造高效商业网站?建站目的决定转化率
北京的网站制作公司有哪些,哪个视频网站最好?
怎么用手机制作网站链接,dw怎么把手机适应页面变成网页?
非常酷的网站设计制作软件,酷培ai教育官方网站?
如何通过智能用户系统一键生成高效建站方案?
济南网站建设制作公司,室内设计网站一般都有哪些功能?
油猴 教程,油猴搜脚本为什么会网页无法显示?
胶州企业网站制作公司,青岛石头网络科技有限公司怎么样?
网站制作价目表怎么做,珍爱网婚介费用多少?
如何用PHP工具快速搭建高效网站?
建站主机系统SEO优化与智能配置核心关键词操作指南
如何在阿里云ECS服务器部署织梦CMS网站?
视频网站制作教程,怎么样制作优酷网的小视频?
如何用免费手机建站系统零基础打造专业网站?
如何快速使用云服务器搭建个人网站?
宝塔建站助手安装配置与建站模板使用全流程解析
如何配置支付宝与微信支付功能?
深圳 网站制作,深圳招聘网站哪个比较好一点啊?
如何通过西部数码建站助手快速创建专业网站?
Android自定义控件实现温度旋转按钮效果
c# await 一个已经完成的Task会发生什么
中山网站制作网页,中山新生登记系统登记流程?
网站制作新手教程,新手建设一个网站需要注意些什么?
建站ABC备案流程中有哪些关键注意事项?
详解免费开源的DotNet二维码操作组件ThoughtWorks.QRCode(.NET组件介绍之四)
如何用wdcp快速搭建高效网站?
香港代理服务器配置指南:高匿IP选择、跨境加速与SEO优化技巧
怎么制作一个起泡网,水泡粪全漏粪育肥舍冬季氨气超过25ppm,可以有哪些措施降低舍内氨气水平?
建站中国官网:模板定制+SEO优化+建站流程一站式指南
北京制作网站的公司,北京铁路集团官方网站?
*请认真填写需求信息,我们会在24小时内与您取得联系。