Redis锁的介绍「建议收藏」

Redis锁的介绍「建议收藏」Redis锁的实现:由于Redis是单进程的,可以简单用setnx这个命令进行加锁操作,谁能操作成功,谁就可以获得锁。简单的代码如下:defacquire_lock():   #identifier:唯一标识客户端   #lockname锁名字   #redis客户端连接   ifredis.setnx(lockname,identifier):     …

大家好,又见面了,我是你们的朋友全栈君。

Redis锁的实现:
由于Redis是单进程的,可以简单用setnx这个命令进行加锁操作,谁能操作成功,谁就可以获得锁。简单的代码如下:
def acquire_lock():
    # identifier: 唯一标识客户端
    # lockname 锁名字
    # redis 客户端连接
    if redis.setnx(lockname, identifier):
        return True
    return False
这里有一个问题,就是如果客户端在获得锁的时候崩溃了,服务器就无法再把锁分配给其他客户端使用了,为了解决这个问题,我们可以利用redis的超时特性,给锁加上超时时间
def acquire_lock():
    # identifier: 唯一标识客户端
    # lockname 锁名字
    # redis 客户端连接
    # timeout 超时时间
    if redis.setnx(lockname, identifier):
        redis.expire(lockname, timeout)
        return True
    elif not redis.ttl(lockname):
        redis.expire(lockname, timeout)
        return False
    return False
可以这样认为,多个客户端同时设置过期时间也是差别不大的,我们在发现锁已经存在并且没有超时限制时,给锁加上超时限制,这样可以在其他客户端获得锁并未设置超时时间崩溃了,也能在过期时间到了让其他客户端获取到锁。最新官方文档支持用set命令指定超时和nx特性,
def acquire_lock():
    # identifier: 唯一标识客户端
    # lockname 锁名字
    # redis 客户端连接
    # timeout 超时时间
    if redis.set(lockname, identifier, nx=True, ex=timeout):
        return True
    return False
解锁操作,直接执行一段lua脚本
def release_lock():
    # identifier: 唯一标识客户端
    # lockname 锁名字
    # redis 客户端连接
    script = “””
        if redis.call(‘GET’, KEYS[1]) == ARGV[1] then
            return redis.call(‘DEL’, KEYS[1])
        else
            return 0
    “””
    return redis.eval(script, lockname, identifier)
使用lua脚本可以原子的操作解锁过程,这里需要注意点,eval的key是要传的,这样代码也可以在redis集群中使用,否则redis不知道lua脚本应该在哪一个槽进行执行,具体可以看官方的文档

另外一种经常使用的计数信号量的使用,最简单的方式是用一个zset,通过客户端设置过期时间
def acquire_semaphore():
    # identifier: 唯一标识客户端
    # semname 信号量集合名
    # redis 客户端连接
    # timeout 过期时间
    # limit 信号量的计数
    now = time.time()
    pipe = redis.pipeline(True)
    pipe.zremrangebyscore(semname, ‘-inf’, now-timeout)
    pipe.zadd(semname, identifier, now)
    pipe.zrank(semname, identifier)
    if pipe.execute()[-1] < limit:
        return True
    redis.zrem(semname, identifier)
    return False
这里有一个问题,就是如果某一个客户端的时间比较快,那么这个较快的客户端是有可能在信号量已经达到限制的时候获得新号量的。
比如A客户端比其他客户端快50ms,那么当最后一个信号量被取走的同时,如果这时候A请求信号量,根据代码是有可以获得信号量的,不是公平的
为了解决公平的问题,可以引入多一个有序集合,记录获取信号量的计数

def acquire_fair_semaphore():
    # identifier: 唯一标识客户端
    # semname 信号量集合名
    # redis 客户端连接
    # timeout 过期时间
    # limit 信号量的计数
    czset = semname + “:owner”
    ctr = semname + “:counter”
    now = time.time()
    pipe = redis.pipeline(True)
    pipe.zramrangebyscore(semname, ‘-inf’, now-timeout)
    pipe.zinterstore(czset, {czset:1, semname: 0})
    pipe.incr(ctr)
    counter = pipe.execute()[-1]
    pipe.zadd(semname, identifier, now)
    pipe.zadd(czset, identifier, counter)
    pipe.zrank(czset, identifier)
    if pipe.execute()[-1] < limit:
        return True
    pipe.zrem(semname, identifier)
    pipe.zrem(czset, identifier)
    pipe.execute()
    return False    
这个看似没啥问题的代码还是有一点问题,假设剩下最后一个信号量,AB两个客户端依次执行incr获得counter = 10,11
这个时候如果B先执行了zrank,那么会导致A和B都同时获得了最后一个信号量,如果场景不在乎这点小问题也没问题,不然可以在这段操作前,先获取前面的锁,再进行操作。

上面介绍的锁其实有一个假设是Redis服务器没有挂,如果Redis服务器挂了,有主从切换的话会切换到从服务器,但是从服务器并不一定与主服务器数据完全一致,取决于同步的方式,假设A获得锁之后服务器挂了,这个信息没有记录到从服务器中,从服务器起来的时候是无锁状态的,有可能会造成两个客户端同时获取了锁,一种解决方法是如果发生主从切换的话,暂停新启动的服务器使用锁的时间,超过过期时间即可。

基于Redis实现的一种分布式锁,RedLock算法
客户端以当前时间毫秒级向多个独立的Redis实例请求锁,超过实例总数(N) N/2+1的客户端获取到这把锁,而释放锁的操作只需要在所有的实例执行释放锁操作。
具体可以参考官方文档:https://redis.io/topics/distlock

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/149192.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • html超链接样式「建议收藏」

    html超链接样式「建议收藏」a:link,定义正常链接的样式;a:visited,定义已访问过链接的样式;a:hover,定义鼠标悬浮在链接上时的样式;a:active,定义鼠标点击链接时的样式。

    2022年7月19日
    24
  • 香港坚固金业的黑幕,属于非法投资平台。

    2015年4.29晚上凌晨两点半我结束了建仓操作,当时没又发现账户资金问题,后来第二天早晨发现账户被非法操作做单,所有资金信息都被在美国时间22:08开始到22:56仅仅不到一个小时时间,所有资金都频繁建仓平仓操作,导致我账户资金全部亏损。 第二天咨询客服客服回答说是我自己认为操作要么,账户信息被泄露,被他人非法操作了。他们查询说是操作IP地址是外地操作,我感觉到很可悲,

    2022年4月4日
    65
  • 如何让phpmyadmin输入密码再进入

    如何让phpmyadmin输入密码再进入

    2021年9月19日
    71
  • 快速阶乘算法python_【最全】阶乘算法!(python和C语言)

    快速阶乘算法python_【最全】阶乘算法!(python和C语言)阶乘的计算叁岁学编程:用最简单的大白话理解编程,欢迎大家关注,留言,提问,希望和大家一起提升!文章目录阶乘的计算阶乘定义:解析方法一:for循环计算方法二:定义for循环的函数计算方法三:定义递归函数计算小知识:C语言代码方法一:for函数方法二:递归函数总结:阶乘定义:阶乘指从1乘以2乘以3乘以4一直乘到所要求的数。例如所要求的数是4,则阶乘式是1×2×3×4,得到的积是24。24就是4的阶乘。…

    2022年7月24日
    10
  • android 4怎么打开usb调试?「建议收藏」

    android 4怎么打开usb调试?「建议收藏」手机连接电脑,刷机,等都需要打开手机USB调试模式,你才能进行操作的。所以买了手机建议都要打开这个USB调试,手机锁屏密码忘记也需要打开这个。这个比较重要。工具/原料手机安卓android系统方法/步骤打开自己的手机找到《设置》点击,进入以下图例。在点击《关于手机》

    2025年11月4日
    4
  • 51单片机系列有哪些类型_51单片机1602液晶显示原理

    51单片机系列有哪些类型_51单片机1602液晶显示原理Lcd液晶屏幕带有背光显示,可选择蓝光白光黄光三种背光颜色的屏幕,它更具有美光性,在实际生活中有广泛的应用。实物图原理图RS:1为数据/0为指令;RW:1为读/0为写;E1为数据有效,下降沿执行指令示例代码main.c#include”reg52.h” //此文件中定义了单片机的一些特殊功能寄存器#include”lcd.h”typedefunsignedintu16; //对数据类型进行声明定义typedefunsignedcharu8;u8Disp

    2025年12月11日
    2

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号