43

Redis应用-分布式锁

 5 years ago
source link: https://www.tuicool.com/articles/RjaeaaF
Go to the source link to view the article. You can view the picture content, updated content and better typesetting reading experience. If the link is broken, please click the button below to view the snapshot at that time.
neoserver,ios ssh client

uiaeM3j.gif

当多个进程不在同一个系统中,就需要用分布式锁控制多个进程对资源的访问。

使用redis来实现分布式锁主要用到以下命令:

  • SETNX KEY VALUE 如果key不存在,就设置key对应字符串value

  • expire KEY seconds 设置key的过期时间

  • del KEY 删除key

代码实现如下:

$redis = new Redis();

$redis->connect('127.0.0.1', 6379);

$ok = $redis->setNX($key, $value);

if ($ok) {

//获取到锁

... do something ...

$redis->del($key);

}

上面代码有没有问题呢? 如果我们在逻辑处理过程中出现了异常情况,导致KEY没有删除,那就出现了死锁了。所以一般我们在拿到锁之后再给KEY加一个过期时间

为了保证执行的原子性,使用了 multi 就有了如下代码

$redis->multi();

$redis->setNX($key, $value);

$redis->expire($key, $ttl);

$res = $redis->exec();

if($res[0]) {

//获取到锁

... do something ...

$redis->del($key);

}

但是这样的又有一个问题第一个请求成功了,之后的请求虽然没有拿到锁但是每次都刷新了锁的时间。这样我们设置锁过期时间的意义就不存在了。所以我们在拿到锁以后再进行过期时间的操作,这时候我们就可以祭出原子性操作的lua脚本,代码如下

$script = <<<EOT

local key = KEYS[1]

local value = KEYS[2]

local ttl = KEYS[3]


local ok = redis.call('setnx', key, value)


if ok == 1 then

redis.call('expire', key, ttl)

end

return ok

EOT;


$res = $redis->eval($script, [$key,$val, $ttl], 3);

if($res) {

//获取到锁

... do something ...

$redis->del($key);

}

借助lua脚本虽然解决了问题,但是未免有些麻烦,Redis从 2.6.12 版本开始, SET 命令的行为可以通过一系列参数来修改:

  • EX second :设置键的过期时间为 second 秒。 SET key value EX second 效果等同于 SETEX key second value 。

  • PX millisecond :设置键的过期时间为 millisecond 毫秒。 SET key value PX millisecond 效果等同于 PSETEX key millisecond value 。

  • NX :只在键不存在时,才对键进行设置操作。 SET key value NX 效果等同于 SETNX key value 。

  • XX :只在键已经存在时,才对键进行设置操作。

$ok = $redis->set($key, $random, array('nx', 'ex' => $ttl));


if ($ok) {

//获取到锁

... do something ...

if ($redis->get($key) == $random) {

$redis->del($key);

}

}

可以看到上面我们我们的值引入了一个随机数,这是为了防止逻辑处理时间过长导致锁的过期时间已经失效,这时候下一个请求就获得了锁,但是前一个请求在逻辑处理完直接删除了锁。

锁主要用在并发请求如秒杀等场景中,以上便是redis锁的实现。

资源

各类学习资源可关注公号获得

good good study  day day up

YJJJJbe.jpg!web

Recommend

  • 102
    • 微信 mp.weixin.qq.com 7 years ago
    • Cache

    浅谈Redis分布式锁实现

    浅谈Redis分布式锁实现 左手...

  • 117
    • blueskykong.com 7 years ago
    • Cache

    基于redis的分布式锁实现

    This blueskykong.com page can’t be found No webpage was found for the web address: http://blueskykong.com/2018/01/06/redislock/?utm_source=tuicool&utm_medium=referral...

  • 66

    分布式锁很久之前有讲过并发编程中的锁并发编程的锁机制:synchronized和lock。在单进程的系统中,当存在多个线程可以同时改变某个变量时,就需要对变量或代码块做同步,使其在修改这种变量时能够线性执行消除并发修改变量。而同步的本质是通过锁来实现的。为了实...

  • 80
    • www.xiongxiangming.com 7 years ago
    • Cache

    分布式锁之redis实现

    摘要:分布式架构设计如今在企业中被大量的应用,而在不同的分布式节点进行协同工作的时候,节点服务的时序、结果的正确性以及执行成本也成为了必须考虑的重要因素。其中竞态条件会导致执行结果的不正确,不同服务节点同时处理同一任务也将耗费不必需的系统资源,...

  • 64

    前言分布式锁在分布式应用中应用广泛,想要搞懂一个新事物首先得了解它的由来,这样才能更加的理解甚至可以举一反三。首先谈到分布式锁自然也就联想到分布式应用。在我们将应用拆分为分布式应用之前的单机系统中,对一些并发场景读取公共资源时如扣库存,卖车票之...

  • 66
    • segmentfault.com 7 years ago
    • Cache

    基于 Redis 的分布式锁

    基于 Redis 的分布式锁

  • 70
    • tech.dianwoda.com 7 years ago
    • Cache

    Redis分布式锁进化史

    Redis分布式锁进化史

  • 58
    • 掘金 juejin.im 6 years ago
    • Cache

    redis系列:分布式锁

    1 介绍 这篇博文讲介绍如何一步步构建一个基于Redis的分布式锁。会从最原始的版本开始,然后根据问题进行调整,最后完成一个较为合理的分布式锁。 本篇文章会将分布式锁的实现分为两部分,一个是单机环境,另一个是集群环境下的Redis锁实现。在介绍分布式锁的实现...

  • 7

    【DB系列】Redis实现分布式锁(应用篇) ...

  • 2
    • blog.51cto.com 3 years ago
    • Cache

    基于Redis分布式BitMap的应用

    在实际开发中常常遇到如下需求:判断当前元素是否存在于已知的集合中,将已知集合中的元素维护一个HashSet,使用时只需耗时O(1)的时间复杂度便可判断出结果,Java内部或者Redis均提供相应的数据结构。使用此种方式除了占用内存空间外...

About Joyk


Aggregate valuable and interesting links.
Joyk means Joy of geeK