Redis锁的使用

起因:分布式环境下需对并发进行逻辑一致性控制

架构:springboot2、Redis

IDEA实操

  1. 先新建RedisLock组件

    注:释放锁使用lua脚本保持原子性

    @Component
    @Slf4j
    public class RedisLock { private final RedisTemplate redisTemplate; public RedisLock(RedisTemplate redisTemplate) {
    this.redisTemplate = redisTemplate;
    } /**
    * 如果已经存在返回false,否则返回true
    *
    * @param key
    * @param value
    * @return
    */
    public Boolean setNx(String key, String value, Long expireTime, TimeUnit mimeUnit) { if (key == null || value == null) {
    return false;
    } // 在spiring boot 2 可以直接使用 redisTemplate的setIfAbsent设置key-value和过期时间,是原子性
    Boolean tf = redisTemplate.opsForValue().setIfAbsent(key, value, expireTime, mimeUnit); return tf; } /**
    * 获取数据
    *
    * @param key
    * @return
    */
    public Object get(String key) { if (key == null) {
    return null;
    }
    return redisTemplate.opsForValue().get(key);
    } /**
    * 删除
    *
    * @param key
    * @return
    */
    public void remove(Object key) { if (key == null) {
    return;
    } redisTemplate.delete(key);
    } /**
    * 加锁
    *
    * @param key key
    * @param waitTime 等待时间,在这个时间内会多次尝试获取锁,超过这个时间还没获得锁,就返回false
    * @param interval 间隔时间,每隔多长时间尝试一次获的锁
    * @param expireTime key的过期时间
    */
    public Boolean lock(String key, Long waitTime, Long interval, Long expireTime) { String value = UUID.randomUUID().toString().replaceAll("-", "").toLowerCase(); Boolean flag = setNx(key, value, expireTime, TimeUnit.MILLISECONDS); // 尝试获取锁 成功返回
    if (flag) {
    return true;
    } else {
    // 获取失败 // 现在时间
    long newTime = System.currentTimeMillis(); // 等待过期时间
    long loseTime = newTime + waitTime; // 不断尝试获取锁成功返回
    while (System.currentTimeMillis() < loseTime) { Boolean testFlag = setNx(key, value, expireTime, TimeUnit.MILLISECONDS);
    if (testFlag) {
    return true;
    } try {
    Thread.sleep(interval);
    } catch (InterruptedException e) {
    log.error("获取锁异常", e);
    }
    }
    }
    return false;
    } /**
    * 释放锁
    *
    * @param key
    * @return
    */
    public void unLock(String key) {
    remove(key);
    } public Boolean setIfAbsent(String key, String value) {
    Boolean tf = redisTemplate.opsForValue().setIfAbsent(key, value);
    redisTemplate.expire(key, 60, TimeUnit.DAYS);
    return tf;
    } /**
    * 获取分布式锁
    * @param key
    * @param value
    * @param expireTime
    * @return
    */
    public Boolean lock(String key, String value, Long expireTime) {
    Boolean flag = setNx(key, value, expireTime, TimeUnit.MILLISECONDS);
    if (flag == null || !flag) {
    return false;
    } return true;
    } /***
    * lua释放分布式锁
    * @param key
    * @param value
    */
    public void unLock(String key, String value) {
    String script = "if redis.call('get', KEYS[1]) == ARGV[1] then return redis.call('del', KEYS[1]) else return 0 end";
    RedisScript<Long> redisScript = new DefaultRedisScript<>(script, Long.class);
    Long result = (Long) redisTemplate.execute(redisScript, Arrays.asList(key),value);
    if (result == null || result == 0) {
    log.info("释放锁(" + key + "," + value + ")失败,该锁不存在或锁已经过期");
    } else {
    log.info("释放锁(" + key + "," + value + ")成功");
    }
    } /***
    * lua不存在才插入队列
    * @param key
    * @param values
    * jackson序列化后String会变成\"xxx\",不要直接用ARGV去转成number使用
    */
    public void putAllIfAbsent(String key, List<String> values,long timeout,TimeUnit unit) {
    Long rawTimeout = TimeoutUtils.toMillis(timeout, unit);
    String script = "if redis.call('exists', KEYS[1]) == 0 then local listValues = ARGV" +
    " for k,v in ipairs(listValues) do " +
    " redis.call('RPUSH',KEYS[1],v) " +
    " end " +
    " redis.call('expire',KEYS[1],KEYS[2])" +
    " end";
    RedisScript<Void> redisScript = new DefaultRedisScript<>(script);
    redisTemplate.execute(redisScript, Arrays.asList(key,String.valueOf(rawTimeout)),values.toArray());
    }
    }
  2. 业务使用

    进入逻辑前先判断有没有锁

    用try-catch包住业务逻辑,finally释放锁,以防抛错直接占有锁

    if (redisLock.lock(redisLockKey, value, 60 * 1000L)) {
    try {
    return reactiveMongoTemplate.find(query, JrRedPacketTask.class)
    .collectList()
    .flatMap(taskList -> {
    if (taskList.size() > 0) {
    //缓存到redis中
    return reactiveRedisTemplate.opsForValue().set(redisKey, taskList, Duration.ofDays(1))
    .then(Mono.just(taskList));
    }
    return Mono.just(taskList);
    });
    } finally {
    redisLock.unLock(redisLockKey, value);// 释放锁
    } } else {
    try {
    //休息,休息一会儿
    Thread.sleep(8);
    } catch (InterruptedException e) {
    e.printStackTrace();
    }
    }

Redis分布式锁应用的更多相关文章

  1. 利用redis分布式锁的功能来实现定时器的分布式

    文章来源于我的 iteye blog http://ak478288.iteye.com/blog/1898190 以前为部门内部开发过一个定时器程序,这个定时器很简单,就是配置quartz,来实现定 ...

  2. Redis分布式锁

    Redis分布式锁 分布式锁是许多环境中非常有用的原语,其中不同的进程必须以相互排斥的方式与共享资源一起运行. 有许多图书馆和博客文章描述了如何使用Redis实现DLM(分布式锁管理器),但是每个库都 ...

  3. redis分布式锁和消息队列

    最近博主在看redis的时候发现了两种redis使用方式,与之前redis作为缓存不同,利用的是redis可设置key的有效时间和redis的BRPOP命令. 分布式锁 由于目前一些编程语言,如PHP ...

  4. redis咋么实现分布式锁,redis分布式锁的实现方式,redis做分布式锁 积极正义的少年

    前言 分布式锁一般有三种实现方式:1. 数据库乐观锁:2. 基于Redis的分布式锁:3. 基于ZooKeeper的分布式锁.本篇博客将介绍第二种方式,基于Redis实现分布式锁.虽然网上已经有各种介 ...

  5. spring boot redis分布式锁

    随着现在分布式架构越来越盛行,在很多场景下需要使用到分布式锁.分布式锁的实现有很多种,比如基于数据库. zookeeper 等,本文主要介绍使用 Redis 做分布式锁的方式,并封装成spring b ...

  6. Redis分布式锁的正确实现方式

    前言 分布式锁一般有三种实现方式:1. 数据库乐观锁:2. 基于Redis的分布式锁:3. 基于ZooKeeper的分布式锁.本篇博客将介绍第二种方式,基于Redis实现分布式锁.虽然网上已经有各种介 ...

  7. Redis分布式锁---完美实现

    这几天在做项目缓存时候,因为是分布式的所以需要加锁,就用到了Redis锁,正好从网上发现两篇非常棒的文章,来和大家分享一下. 第一篇是简单完美的实现,第二篇是用到的Redisson. Redis分布式 ...

  8. redis分布式锁实践

    分布式锁在多实例部署,分布式系统中经常会使用到,这是因为基于jvm的锁无法满足多实例中锁的需求,本篇将讲下redis如何通过Lua脚本实现分布式锁,不同于网上的redission,完全是手动实现的 我 ...

  9. Redis分布式锁的try-with-resources实现

    Redis分布式锁的try-with-resources实现 一.简介 在当今这个时代,单体应用(standalone)已经很少了,java提供的synchronized已经不能满足需求,大家自然 而 ...

  10. 关于分布式锁原理的一些学习与思考-redis分布式锁,zookeeper分布式锁

    首先分布式锁和我们平常讲到的锁原理基本一样,目的就是确保,在多个线程并发时,只有一个线程在同一刻操作这个业务或者说方法.变量. 在一个进程中,也就是一个jvm 或者说应用中,我们很容易去处理控制,在j ...

随机推荐

  1. @property装饰器和property()函数

    @property装饰器 Python内置的@property装饰器可以把类的方法伪装成属性调用的方式.也就是本来是Foo.func()的调用方法,变成Foo.func的方式. class Peopl ...

  2. 8. 使用Fluentd+MongoDB采集Apache日志

    Fluentd+MongoDB,用以实时收集半结构化数据. 背景知识 日志接入Fluentd后,会以json的格式在Fluentd内部进行路由.这就决定了Fluentd处理日志的方式是非常灵活的,它将 ...

  3. 【golang】json数据解析 - 嵌套json解析

    @ 目录 1. 通过结构体映射解析 2. 嵌套json解析-map 1. 通过结构体映射解析 原数据结构 解析 // 结构体 type contractJson struct { Data []tra ...

  4. Anaconda安装和卸载+虚拟环境Tensorflow安装以及末尾问题大全(附Anaconda安装包),这一篇就够了!!!

    前言 实话说,在自己亲手捣鼓了一下午加一晚上后,本人深深地感受到了对于"Anaconda安装+虚拟环境Tensorflow安装"里面的坑点之多,再加上目前一些博主的资料有点久远,尤 ...

  5. linux 安装/卸载go环境

    linux 安装/卸载go环境(基于centos8) 安装 下载go的安装包 Golang官网下载地址:https://golang.org/dl/ 将安装包解压放到到usr/local中,并解压 c ...

  6. Pytest进阶使用

    fixture 特点: 命令灵活:对于setup,teardown可以省略 数据共享:在conftest.py配置里写方法可以实现数据共享,不需要import导入,可以跨文件共享 scope的层次及神 ...

  7. vue+spirngboot 分离技术实现图书信息的增删改查(改造这学期的课程设计【1】)

    1.前端项目的创建 vue init webpack bookshopvue 安装axios http://www.axios-js.com/ npm install --save axios vue ...

  8. 【React】学习笔记(二)——组件的生命周期、React脚手架使用

    原教程视频:ttps://www.bilibili.com/video/BV1wy4y1D7JT?p=2&spm_id_from=pageDriver 目录 一.组件的生命周期 1.1.生命周 ...

  9. Spring Retry 重试

    重试的使用场景比较多,比如调用远程服务时,由于网络或者服务端响应慢导致调用超时,此时可以多重试几次.用定时任务也可以实现重试的效果,但比较麻烦,用Spring Retry的话一个注解搞定所有.话不多说 ...

  10. 某 .NET RabbitMQ SDK 有采集行为,你怎么看?

    一:背景 1.讲故事 前几天有位朋友在微信上找到我,说他的一个程序上了生产之后,被运维监控定位到这个程序会向一个网址为: http://m.365ey.net 上不定期打数据,而且还是加密的格式,要他 ...