在分布式系统中,经常会出现需要竞争同一资源的情况,使用redis可以实现分布式锁。

前提:redis集群已经整合项目,并且可以直接注入JedisCluster使用:

  1. @Autowired
  2. private JedisCluster jedisCluster;

1. 新建RedisLockManger分布式锁管理器,并且如上注入 JedisCluster :

  1. package com.jarfk.util.redis;
  2.  
  3. import org.slf4j.Logger;
  4. import org.slf4j.LoggerFactory;
  5. import org.springframework.beans.factory.annotation.Autowired;
  6. import org.springframework.stereotype.Component;
  7. import redis.clients.jedis.JedisCluster;
  8.  
  9. import javax.annotation.PostConstruct;
  10. import java.util.concurrent.TimeUnit;
  11.  
  12. /**
  13. * redis集群分布式锁管理器,支持对单个资源加锁解锁,或给一批资源的批量加锁及解锁
  14. * Created by Administrator on 2017/10/12 0012.
  15. */
  16. @Component
  17. public class RedisLockManger {
  18. private static final Logger LOGGER = LoggerFactory.getLogger(RedisLockManger.class);
  19.  
  20. //设置3秒过期
  21. private static final int DEFAULT_SINGLE_EXPIRE_TIME = 3;
  22.  
  23. // private static final int DEFAULT_BATCH_EXPIRE_TIME = 6;
  24.  
  25. //static的变量无法注解
  26. @Autowired
  27. private JedisCluster jc;
  28.  
  29. private static RedisLockManger lockManger;
  30.  
  31. public RedisLockManger() {
  32. }
  33.  
  34. @PostConstruct
  35. private void init() {
  36. lockManger = this;
  37. lockManger.jc = this.jc;
  38. }
  39. /**
  40. * 获取锁 如果锁可用 立即返回true, 否则立即返回false,作为非阻塞式锁使用
  41. * @param key
  42. * @return
  43. */
  44. public boolean tryLock(String key/* , String value*/) {
  45. try {
  46. return tryLock(key, key, 0L, null);
  47. } catch (InterruptedException e) {
  48. e.printStackTrace();
  49. }
  50. return false;
  51. }
  52.  
  53. /**
  54. * 锁在给定的等待时间内空闲,则获取锁成功 返回true, 否则返回false,作为阻塞式锁使用
  55. * @param key 锁键
  56. * @param value 被谁锁定
  57. * @param timeout 尝试获取锁时长,建议传递500,结合实践单位,则可表示500毫秒
  58. * @param unit,建议传递TimeUnit.MILLISECONDS
  59. * @return
  60. * @throws InterruptedException
  61. */
  62. public boolean tryLock(String key , String value , long timeout , TimeUnit unit) throws InterruptedException {
  63. //纳秒
  64. long begin = System.nanoTime();
  65. do {
  66. //LOGGER.debug("{}尝试获得{}的锁.", value, key);
  67. Long i = lockManger.jc.setnx(key, value);
  68. if (i == 1) {
  69. lockManger.jc.expire(key, DEFAULT_SINGLE_EXPIRE_TIME);
  70. LOGGER.debug(value + "-成功获取{}的锁,设置锁过期时间为{}秒 ", key, DEFAULT_SINGLE_EXPIRE_TIME);
  71. return true;
  72. } else {
  73. // 存在锁 ,但可能获取不到,原因是获取的一刹那间
  74. // String desc = lockManger.jc.get(key);
  75. // LOGGER.error("{}正被{}锁定.", key, desc);
  76. } if (timeout == 0) {
  77. break;
  78. }
  79. //在其睡眠的期间,锁可能被解,也可能又被他人占用,但会尝试继续获取锁直到指定的时间
  80. Thread.sleep(100);
  81. } while ((System.nanoTime() - begin) < unit.toNanos(timeout));
  82. //因超时没有获得锁
  83. return false;
  84. }
  85.  
  86. /**
  87. * 释放单个锁
  88. * @param key 锁键
  89. */
  90. public void unLock(String key/*, String value*/) {
  91. lockManger.jc.del(key);
  92. LOGGER.debug("{}锁被{}释放 .", key, key);
  93. }
  94. }

2. 使用示例:

首先在需要加锁的地方注入分布式锁管理器:

  1. @Autowired
  2. private RedisLockManger redisLock;

然后调用即可,如:

  1. if (redisLock.tryLock("statusCheck")) { //此处代码是锁上的
  2. logger.debug("-----------------------:10秒执行一次!每次只有一个程序运行");
  3. //释放锁,正常情况下,此处代码要注释掉,以免锁被释放,需要释放时可以根据自己逻辑的需要
  4. //redisLock.unLock("statusCheck");
  5. }

首先注入需要的

基于redis集群实现的分布式锁,可用于秒杀,定时器。的更多相关文章

  1. 基于redis集群实现的分布式锁,可用于秒杀商品的库存数量管理,有測试代码(何志雄)

    转载请标明出处. 在分布式系统中,常常会出现须要竞争同一资源的情况,本代码基于redis3.0.1+jedis2.7.1实现了分布式锁. redis集群的搭建,请见我的另外一篇文章:<>& ...

  2. 分布式ID系列(4)——Redis集群实现的分布式ID适合做分布式ID吗

    首先是项目地址: https://github.com/maqiankun/distributed-id-redis-generator 关于Redis集群生成分布式ID,这里要先了解redis使用l ...

  3. springcloud微服务基于redis集群的单点登录

    springcloud微服务基于redis集群的单点登录 yls 2019-9-23 简介 本文介绍微服务架构中如何实现单点登录功能 创建三个服务: 操作redis集群的服务,用于多个服务之间共享数据 ...

  4. 【spring boot】【redis】spring boot基于redis的LUA脚本 实现分布式锁

    spring boot基于redis的LUA脚本 实现分布式锁[都是基于redis单点下] 一.spring boot 1.5.X 基于redis 的 lua脚本实现分布式锁 1.pom.xml &l ...

  5. 集群多JVM分布式锁实现

    基于数据库表乐观锁 (基本废弃) 要实现分布式锁,最简单的⽅方式可能就是直接创建⼀一张锁表,然后通过操作该表中的数据来实现了了. 当我们要锁住某个⽅法或资源时,我们就在该表中增加一条记录,想要释放锁的 ...

  6. Redis集群模式之分布式集群模式

    前言 Redis集群模式主要有2种: 主从集群 分布式集群. 前者主要是为了高可用或是读写分离,后者为了更好的存储数据,负载均衡. 本文主要讲解主从集群.本章主要讲解后一半部分,Redis集群. 与本 ...

  7. 【连载】redis库存操作,分布式锁的四种实现方式[四]--基于Redis lua脚本机制实现分布式锁

    一.redis lua介绍 Redis 提供了非常丰富的指令集,但是用户依然不满足,希望可以自定义扩充若干指令来完成一些特定领域的问题.Redis 为这样的用户场景提供了 lua 脚本支持,用户可以向 ...

  8. java-spring基于redis单机版(redisTemplate)实现的分布式锁+redis消息队列,可用于秒杀,定时器,高并发,抢购

    此教程不涉及整合spring整合redis,可另行查阅资料教程. 代码: RedisLock package com.cashloan.analytics.utils; import org.slf4 ...

  9. Couchbase集群和Redis集群解析

    Couchbase集群和Redis集群解析 首先,关于一些数据库或者是缓存的集群有两种结构,一种是Cluster;一种是master-salve. 关于缓存系统一般使用的就是Redis,Redis是开 ...

随机推荐

  1. stopManagedWebLogic.sh强制关闭Managed Server

    Adding force shutdown of managed server in weblogic. ----------------------------------------------- ...

  2. Cents os 7下如何安装bzip2

    # Cents os 7下如何安装bzip2 ### 安装```yum search bzip2  //查询安装包 yum -y install bzip2.x86_64 ``` ### 原因---- ...

  3. Oracle用户密码过期的处理方法

    受影响版本:Oracle11g以上版本.   导致密码消失的原因:Oracle 11g中默认的DEFAULT概要文件中口令有效期PASSWORD_LIFE_TIME默认值为180天.   当以客户端登 ...

  4. 2. 知识图谱-命名实体识别(NER)详解

    1. 通俗易懂解释知识图谱(Knowledge Graph) 2. 知识图谱-命名实体识别(NER)详解 3. 哈工大LTP解析 1. 前言 在解了知识图谱的全貌之后,我们现在慢慢的开始深入的学习知识 ...

  5. 基于jQuery实现的腾讯互动娱乐网站特效

    分享一款基于jQuery实现的腾讯互动娱乐网站特效.腾讯互动娱乐网站jQuery特效是一款右侧带伸缩选项卡,支持鼠标滚轮滚动切换特效代码.效果图如下: 在线预览   源码下载 实现的代码. html代 ...

  6. wcf中的Message类

    客户端->服务端—>客户端 客户端代码: using (new OperationContextScope(client.InnerChannel))            {       ...

  7. [转]基于BootStrap 的城市三级联动

    原文地址:https://blog.csdn.net/peiyuanxin/article/details/51992384 HTML代码部分    <div class="form- ...

  8. Swift is Open Source 博客note

    Swift is Open Sourcehtml, body {overflow-x: initial !important;}html { font-size: 14px; } body { mar ...

  9. composer安装与应用

    操作环境:centos 6.5+32bit 1. 建立项目目录 mkdir test cd test 2. 在当前目录下安装: $ curl -sS https://getcomposer.org/i ...

  10. Domain应用之 根据某个Many2one的对象的 X2many对象 过滤

    如果两者都是many2one类型的对象,过滤非常简单,在xml中添加domain过滤即可,比如 国家.省市之间的联动关系. 如果想要根据某个对象的X2many类型的字段进行过滤该如何去做呢? 答案是利 ...