热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

基于Redis实现分布式锁剖析

之前的文章《分布式锁详解-分别利用Zookeeper和数据库实现分布式锁》,由于篇幅太长,又碰上加班时间不够充裕,所以没有把Redis的实

之前的文章《分布式锁详解 - 分别利用Zookeeper和数据库实现分布式锁》,由于篇幅太长,又碰上加班时间不够充裕,所以没有把Redis的实现也顺带进去,特此做一些利用Redis实现分布式锁的分析。

PS:让我做选择的话,分布式锁的选择还是更愿意选择利用Zookeeper去实现的。
原因的话可以参考下面这张图:在这里插入图片描述


一、关于Redis分布式锁的基础知识


1、缓存有效期:

redis中的数据,不一定都是持久化的;为存储元素设置生存时间,当元素过期时,会被自动删除。(主要是为了防止死锁,一般生存时间设定为业务操作时间的两倍基本就可以了)


2、SETNX命令:

SETNX key value,将key的值设为value,当且仅当key不存在。若给定的key已经存在,则 SETNX 不做任何动作。SETNX是 【 SET if Not eXists 】的简写。(SETNX 命令就是用来保证设置失效时间和插入值是原子操作,如果不是原子执行就可能设置完key,在设置过期时间的时候系统挂机,就会导致死锁问题)


3、lua脚本

轻量小巧的脚本语言,用于支持redis操作序列的原子性。



PS:如果对Redis不是很熟悉的话,建议还是要去学习一下的,现在基本大部分的互联网公司都离不开Redis的掌握。此处顺便分享一个Redis的学习路线图给大家。
在这里插入图片描述


二、Redis实现分布式锁

还是和上一篇文章一样,Redis实现抽象锁AbstractLock,然后拓展其抽象方法。
AbstractLock.java

import java.util.concurrent.locks.Lock;/*** @author WangCw* @create 2019-04-22 19:52* @description**/
public abstract class AbstractLock implements Lock {public void getLock(){//尝试获取锁资源if(tryLock()){System.out.println("获取Lock锁的资源 #####");} else {//等待waitLock();//重新获取锁资源(等待后递归获取锁)getLock();}}//获取锁资源public abstract boolean tryLock();//等待public abstract void waitLock();//释放锁public abstract void unLock();}

1、项目引入Redis相关依赖信息:spring-data-redis(Spring整合Redis一些信息)和jedis(Redis客户端)。

2、Jedis连接池的配置:
在这里插入图片描述
3、实现:

class RedisLock extends AbstractLock{//随机值&#xff0c;保证每个线程随机值独立ThreadLocal<String> local &#61; new ThreadLocal<>();//获取锁的方法 ---> 插入一个元素&#xff0c;成功返回true&#xff0c;否则falsepublic boolean tryLock(){//产生随机值String uuid &#61; UUID.randomUUID.toString();local.set(uuid);//获取redis的原始链接Jedis jedis &#61; (Jedis) factory.getConnection().getNativeConnection();//使用setNx命令请求写值&#xff0c;并设置失效时间String ret &#61; jedis.set(KEY, uuid, "NX", "PX", 1000);//返回"OK"意味着加锁成功if("OK".equals(ret)){return true}return false; }// 等待获取锁的方法&#xff0c;一段缓存时间&#xff0c;防止系统频繁去获取锁public void waitLock() throw Exception{Thread.sleep(3000);}//释放锁 ---> 借助lua脚本删除加锁时添加的元素&#xff0c;保证操作原子性public void unLock() throw Exception{//读取lua脚本String script &#61; FileUtils.readFileByLines("C://unlock.lua");//获取redis的原始链接Jedis jedis &#61; (Jedis) factory.getConnection().getNativeConnection();//通过jedis去执行Lua脚本jedis.eval(script, Array.asList(KEY), Arrays.asList(local.get()));}}

其中上面的lua脚本如下即可&#xff1a;

if redis.call("get", KEYS[1]) &#61;&#61; ARGV[1] thenreturn redis.call("del", KEYS[1])
elsereturn 0
end


在这里插入图片描述


Redisson分布式锁

上面实现的Redis分布式存在一个大问题&#xff1a;假设设置失效时间10秒&#xff0c;如果由于某些原因导致10秒还没执行完任务&#xff0c;这时候锁自动失效&#xff0c;导致其他线程也会拿到分布式锁。面对这种情况&#xff0c;就需要通过一个守护线程&#xff0c;定时检测持有锁的线程是否还在正常做业务&#xff0c;如果是就给分布式锁的过期时间进行续命。实现起来相当复杂。这个时候&#xff0c;就可以借助Redisson

Redisson是一个在Redis的基础上实现的&#xff0c;基于Netty封装了一些利用了Redis特性的工具&#xff0c;让我们可以更高效、更简便的使用Redis。

业界大部分基于Redis的分布式锁都是由Redisson实现的&#xff0c;看看他的使用吧&#xff1a;

RLock lock &#61; redisson.getLock("lock-xxxx");
try{加锁方式一&#xff1a;最常见的使用方法&#xff0c;默认30秒过期&#xff0c;watch dog(看门狗)每过10秒检测一次线程情况&#xff0c;进行续锁lock.lock();加锁方式二&#xff1a;支持过期解锁功能,10秒钟以后自动解锁, 无需调用&#96;unlock()&#96;方法手动解锁lock.lock(10, TimeUnit.SECONDS);加锁方式三&#xff1a;尝试加锁&#xff0c;最多等待3秒&#xff0c;上锁以后10秒自动解锁。boolean res &#61; lock.tryLock(3, 10, TimeUnit.SECONDS);if(res){// 加锁成功&#xff0c;do your business}
finally {解锁方式&#xff1a;lock.unlock();
}

是不是非常简单&#xff0c;直接通过引入Redisson&#xff0c;获取到Rlock&#xff0c;剩下的操作就跟我们Java单机锁的使用方式基本相差无几了。

在这里插入图片描述


Redisson的两个特性&#xff1a;

1、加锁机制


  • 线程去获取锁&#xff0c;获取成功: 执行lua脚本&#xff0c;保存数据到redis数据库。
  • 线程去获取锁&#xff0c;获取失败: 一直通过while循环尝试获取锁&#xff0c;获取成功后&#xff0c;执行lua脚本&#xff0c;保存数据到redis数据库。

2、watch dog自动延期机制
当我们使用lock.lock();方法进行加锁时&#xff0c;默认锁的过期时间是30秒。Redisson会启动一个watch dog(看门狗)后台线程&#xff0c;以锁超时时间 / 3 &#61; 30/3&#61;10s作为周期&#xff08;每过10秒检测一次&#xff09;&#xff0c;检测持有锁的线程是否还在执行业务&#xff0c;还在执行&#xff0c;就进行锁的续期。也就是说&#xff0c;如果一个拿到锁的线程一直没有完成业务逻辑&#xff0c;那么看门狗会帮助线程不断的延长锁超时时间&#xff0c;锁不会因为超时而被释放。默认情况下&#xff0c;看门狗的续期时间是30s&#xff0c;也可以通过修改Config.lockWatchdogTimeout来另行指定。


推荐阅读
  • 分布式开源任务调度框架 TBSchedule 深度解析与应用实践
    本文深入解析了分布式开源任务调度框架 TBSchedule 的核心原理与应用场景,并通过实际案例详细介绍了其部署与使用方法。首先,从源码下载开始,详细阐述了 TBSchedule 的安装步骤和配置要点。接着,探讨了该框架在大规模分布式环境中的性能优化策略,以及如何通过灵活的任务调度机制提升系统效率。最后,结合具体实例,展示了 TBSchedule 在实际项目中的应用效果,为开发者提供了宝贵的实践经验。 ... [详细]
  • Redis:缓存与内存数据库详解
    本文介绍了数据库的基本分类,重点探讨了关系型与非关系型数据库的区别,并详细解析了Redis作为非关系型数据库的特点、工作模式、优点及持久化机制。 ... [详细]
  • 整理于2020年10月下旬:总结过去,展望未来Itistoughtodayandtomorrowwillbetougher.butthedayaftertomorrowisbeau ... [详细]
  • Spring Boot + RabbitMQ 消息确认机制详解
    本文详细介绍如何在 Spring Boot 项目中使用 RabbitMQ 的消息确认机制,包括消息发送确认和消息接收确认,帮助开发者解决在实际操作中可能遇到的问题。 ... [详细]
  • Redis 是一个高性能的开源键值存储系统,支持多种数据结构。本文将详细介绍 Redis 中的六种底层数据结构及其在对象系统中的应用,包括字符串对象、列表对象、哈希对象、集合对象和有序集合对象。通过12张图解,帮助读者全面理解 Redis 的数据结构和对象系统。 ... [详细]
  • Python 数据可视化实战指南
    本文详细介绍如何使用 Python 进行数据可视化,涵盖从环境搭建到具体实例的全过程。 ... [详细]
  • 秒建一个后台管理系统?用这5个开源免费的Java项目就够了
    秒建一个后台管理系统?用这5个开源免费的Java项目就够了 ... [详细]
  • 对象存储与块存储、文件存储等对比
    看到一篇文档,讲对象存储,好奇,搜索文章,摘抄,学习记录!背景:传统存储在面对海量非结构化数据时,在存储、分享与容灾上面临很大的挑战,主要表现在以下几个方面:传统存储并非为非结 ... [详细]
  • 【转】强大的矩阵奇异值分解(SVD)及其应用
    在工程实践中,经常要对大矩阵进行计算,除了使用分布式处理方法以外,就是通过理论方法,对矩阵降维。一下文章,我在 ... [详细]
  • Python学习day3网络基础之网络协议篇
    一、互联网协议连接两台计算机之间的Internet实际上就是一系列统一的标准,这些标准称之为互联网协议,互联网的本质就是一系列网络协议。二、为什么要有互联网协议互联网协议就相当于计 ... [详细]
  • 2020年9月15日,Oracle正式发布了最新的JDK 15版本。本次更新带来了许多新特性,包括隐藏类、EdDSA签名算法、模式匹配、记录类、封闭类和文本块等。 ... [详细]
  • 在CentOS 7环境中安装配置Redis及使用Redis Desktop Manager连接时的注意事项与技巧
    在 CentOS 7 环境中安装和配置 Redis 时,需要注意一些关键步骤和最佳实践。本文详细介绍了从安装 Redis 到配置其基本参数的全过程,并提供了使用 Redis Desktop Manager 连接 Redis 服务器的技巧和注意事项。此外,还探讨了如何优化性能和确保数据安全,帮助用户在生产环境中高效地管理和使用 Redis。 ... [详细]
  • 服务器部署中的安全策略实践与优化
    服务器部署中的安全策略实践与优化 ... [详细]
  • 本文探讨了 Kafka 集群的高效部署与优化策略。首先介绍了 Kafka 的下载与安装步骤,包括从官方网站获取最新版本的压缩包并进行解压。随后详细讨论了集群配置的最佳实践,涵盖节点选择、网络优化和性能调优等方面,旨在提升系统的稳定性和处理能力。此外,还提供了常见的故障排查方法和监控方案,帮助运维人员更好地管理和维护 Kafka 集群。 ... [详细]
  • HBase Java API 进阶:过滤器详解与应用实例
    本文详细探讨了HBase 1.2.6版本中Java API的高级应用,重点介绍了过滤器的使用方法和实际案例。首先,文章对几种常见的HBase过滤器进行了概述,包括列前缀过滤器(ColumnPrefixFilter)和时间戳过滤器(TimestampsFilter)。此外,还详细讲解了分页过滤器(PageFilter)的实现原理及其在大数据查询中的应用场景。通过具体的代码示例,读者可以更好地理解和掌握这些过滤器的使用技巧,从而提高数据处理的效率和灵活性。 ... [详细]
author-avatar
奋怒的小超_656
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有