0

0

Redis实现分布式锁详解

WBOY

WBOY

发布时间:2023-06-21 11:02:38

|

1815人浏览过

|

来源于php中文网

原创

随着移动互联网的快速发展和数据量的爆炸式增长,分布式系统变得越来越普及。分布式系统中,并发操作的问题就变得越来越凸显,当多个线程同时请求共享资源时,就需要对这些资源进行加锁,保证数据的一致性。分布式锁是一种实现分布式系统并发操作的有效方案之一,本文将详细介绍如何使用 redis 实现分布式锁。

  1. Redis 基础

Redis 是一个基于内存的键值对存储系统,在分布式系统中被广泛使用。Redis 作为一种高性能的 NoSQL 数据库,以其高效的读写性能和丰富的数据结构而受到广泛关注。Redis 可以基于多个机器实现分布式存储,同时支持如下数据结构:

  • 字符串(string)
  • 哈希(hash)
  • 列表(list)
  • 集合(set)
  • 有序集合(sorted set)

Redis 的操作都是基于这些数据结构,为实现分布式锁需要用到 Redis 的一个特性:SETNX(SET if Not eXists),即当指定的键不存在时,才能设置键的值。如果键已经存在,则 SETNX 操作会返回失败。

  1. 实现分布式锁的思路

要实现分布式锁,首先需要明确目标:

  • 在分布式环境中,多个线程同时请求同一个资源时,要保证只有一个线程可以获得锁。
  • 如果某个线程已经获得锁,其他线程则需要等待锁的释放。

为了实现上述目标,可以采用以下思路:

  • 使用 Redis 的 SETNX 命令创建一个新的键,作为锁的标识。
  • 如果 SETNX 命令返回成功,表示当前线程获得了锁。
  • 设置键的过期时间,避免死锁的情况。
  • 当某个线程完成任务后,释放锁,即删除该键。
  1. 实现代码示例

首先,创建一个 Redis 连接:

import redis

conn = redis.Redis(host='localhost', port=6379, db=0)

接着,定义获取锁和释放锁的函数:

def acquire_lock(conn, lockname, acquire_timeout=10, lock_timeout=10):
    identifier = str(uuid.uuid4())
    lockname = "lock:" + lockname
    end = time.time() + acquire_timeout
    while time.time() < end:
        if conn.setnx(lockname, identifier):
            conn.expire(lockname, lock_timeout)
            return identifier
        elif not conn.ttl(lockname):
            conn.expire(lockname, lock_timeout)
        time.sleep(0.001)
    return False

def release_lock(conn, lockname, identifier):
    pipe = conn.pipeline(True)
    lockname = "lock:" + lockname
    while True:
        try:
            pipe.watch(lockname)
            if pipe.get(lockname) == identifier:
                pipe.multi()
                pipe.delete(lockname)
                pipe.execute()
                return True
            pipe.unwatch()
            break
        except redis.exceptions.WatchError:
            pass
    return False

其中,acquire_lock 函数用于获取锁,参数说明如下:

Mall4j商城系统
Mall4j商城系统

Mall4j是一个基于spring boot、spring oauth2.0、mybatis、redis的轻量级、前后端分离、防范xss攻击、拥有分布式锁、为生产环境多实例完全准备、数据库为b2b2c设计、拥有完整sku和下单流程的java开源商城。

下载
  • conn:Redis 连接。
  • lockname:锁的名称。
  • acquire_timeout:获取锁时的超时时间,默认为 10 秒。
  • lock_timeout:锁的过期时间,默认为 10 秒。

该函数首先生成一个随机的标识符,然后每隔 0.001 秒尝试获取锁,并设置过期时间。如果在指定的超时时间内没有获取到锁,则返回 False。

release_lock 函数用于释放锁,参数说明如下:

  • conn:Redis 连接。
  • lockname:锁的名称。
  • identifier:获取锁时返回的标识符。

该函数首先使用 WATCH 命令监视锁,如果锁的值与标识符相同,则使用 MULTI 命令删除该锁,并执行操作。否则,终止监视并返回 False。

最后,使用 acquire_lock 和 release_lock 函数即可实现分布式锁的功能。示例代码如下:

import time
import uuid

def do_task():
    print("Task started...")
    time.sleep(5)
    print("Task finished")

def main():
    lockname = "mylock"
    identifier = acquire_lock(conn, lockname)
    if not identifier:
        print("Failed to obtain lock")
        return
    try:
        do_task()
    finally:
        release_lock(conn, lockname, identifier)

if __name__ == '__main__':
    main()

该示例代码中,使用 acquire_lock 函数获取锁,在执行任务后调用 release_lock 函数释放锁。

  1. 总结

分布式锁是一种广泛应用于分布式系统的技术,它可以有效地解决并发操作下数据一致性的问题。在这篇文章中,我们详细介绍了如何使用 Redis 实现分布式锁,通过使用 Redis 的 SETNX 命令和过期时间设置,以及 WATCH 和 MULTI 命令,就可以实现分布式锁的功能。

相关专题

更多
什么是分布式
什么是分布式

分布式是一种计算和数据处理的方式,将计算任务或数据分散到多个计算机或节点中进行处理。本专题为大家提供分布式相关的文章、下载、课程内容,供大家免费下载体验。

319

2023.08.11

分布式和微服务的区别
分布式和微服务的区别

分布式和微服务的区别在定义和概念、设计思想、粒度和复杂性、服务边界和自治性、技术栈和部署方式等。本专题为大家提供分布式和微服务相关的文章、下载、课程内容,供大家免费下载体验。

228

2023.10.07

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

312

2023.08.02

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

712

2023.08.22

mysql标识符无效错误怎么解决
mysql标识符无效错误怎么解决

mysql标识符无效错误的解决办法:1、检查标识符是否被其他表或数据库使用;2、检查标识符是否包含特殊字符;3、使用引号包裹标识符;4、使用反引号包裹标识符;5、检查MySQL的配置文件等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

179

2023.12.04

Python标识符有哪些
Python标识符有哪些

Python标识符有变量标识符、函数标识符、类标识符、模块标识符、下划线开头的标识符、双下划线开头、双下划线结尾的标识符、整型标识符、浮点型标识符等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

271

2024.02.23

java标识符合集
java标识符合集

本专题整合了java标识符相关内容,想了解更多详细内容,请阅读下面的文章。

250

2025.06.11

c++标识符介绍
c++标识符介绍

本专题整合了c++标识符相关内容,阅读专题下面的文章了解更多详细内容。

121

2025.08.07

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

2

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
进程与SOCKET
进程与SOCKET

共6课时 | 0.3万人学习

Redis+MySQL数据库面试教程
Redis+MySQL数据库面试教程

共72课时 | 6.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号