详解RedisTemplate下Redis分布式锁引发的系列问题


Posted in Redis onApril 27, 2021

      自己的项目因为会一直抓取某些信息,但是本地会和线上经常一起跑,造成冲突。这其实就是我们常说的分布式集群的问题了,本地和线上的服务器构成了集群以及QPS为2的小并发(其实也不叫并发,不知道拿什么词形容?)。

     首先,分布式集群的问题大家都知道,会造成数据库的插入重复问题,会造成一系列的并发性问题。

     解决的方式呢也大概如下几点,百度以及谷歌上都能搜到的解决方式:

     1:数据库添加唯一索引

     2:设计接口幂等性

     3:依靠中间件使用分布式锁,而分布式锁又分为Redis和Zookeeper

    由于Zookeeper我没怎么接触过,并且我项目中本来就引用了Redis,所以就想着用Redis来做分布式锁,也高端洋气上档次点。

    首先基于Redis的操作,我们必须要保证其原子性,也就是要么全部成功,要么全部失败,先从Redis的客户端入手。

    就Redis客户端而言,我们通过的操作是先使用setnx指令,如果成功则返回1,失败则返回0

详解RedisTemplate下Redis分布式锁引发的系列问题

   可是就分布锁锁而言,一个常用的问题就是如果一个服务setnx成功了,但是在解锁的时候如果发生了宕机或者一些特殊因素,导致无法解锁,那么其他服务将陷入死锁的状态。所以,我们在用 setnx 的同时想着去用 expire 指令对锁进行一个过期操作

 详解RedisTemplate下Redis分布式锁引发的系列问题

   从指令可以看出 setnx 和 expire 指令是分开的,如果在这中间的空隙过程中如果有特殊因素导致指令无法继续,也会导致死锁的产生。

以下参考自老钱的 Redis 深度历险:核心原理与应用实践

   为了解决这个疑难,Redis 开源社区涌现了一堆分布式锁的 library,专门用来解决这个问题。实现方法极为复杂,小白用户一般要费很大的精力才可以搞懂。如果你需要使用分布式锁,意味着你不能仅仅使用 Jedis 或者 redis-py 就行了,还得引入分布式锁的 library。

  为了治理这个乱象,Redis 2.8 版本中作者加入了 set 指令的扩展参数,使得 setnx 和 expire 指令可以一起执行,彻底解决了分布式锁的乱象。从此以后所有的第三方分布式锁 library 可以休息了。

   详解RedisTemplate下Redis分布式锁引发的系列问题

  以上都是基于Redis的操作,但是我们在JAVA中如何去运用分布式锁呢。

  首先在Redis方面我用的是RedisTemplate对Redis进行操作的 ,而RedisTemplate在目前情况下如果不借助于是无法保证其原子性的,所以我们需要借助于Redis的Lua脚本。

   先上Lua脚本的代码

// 加锁
if 
    redis.call('setNx',KEYS[1],ARGV[1]) 
  then 
    if redis.call('get',KEYS[1])==ARGV[1] 
    return redis.call('expire',KEYS[1],ARGV[2]) 
  else 
    return 0 
  end 
end
 
// 解锁
  redis.call('get', KEYS[1]) == ARGV[1] 
then 
  return redis.call('del', KEYS[1]) 
else 
  return 0

    Java调用脚本有两种方式

   1。新建一个脚本文件,在代码中调用其绝对路径地址

     redisScript.setScriptSource(new ResourceScriptSource(new ClassPathResource(地址)));

   2。在Java代码中以字符串的方式传入

    redisScript.setScriptText(脚本);

  我是用的第二种方式实现的,下面是JAVA代码

/**
   * 获取锁
   * @param lockKey
   * @param value
   * @param expireTime:单位-秒
   * @return
   */
  public boolean getLock(String lockKey, String value, int expireTime){
    boolean ret = false;
    try{
      String script = "if redis.call('setNx',KEYS[1],ARGV[1]) then if redis.call('get',KEYS[1])==ARGV[1] then return redis.call('expire',KEYS[1],ARGV[2]) else return 0 end end";
 
      RedisScript<Long> redisScript = new DefaultRedisScript<>(script, Long.class);
 
      Object result = redisTemplate.execute(redisScript,new StringRedisSerializer(),new StringRedisSerializer(), Collections.singletonList(lockKey),value,expireTime + "");
      System.out.println(result + "-----------");
      //Object result = redisTemplate.execute(redisScript, Collections.singletonList(lockKey),value,expireTime + "");
 
      if(SUCCESS.equals(result)){
        return true;
      }
 
    }catch(Exception e){
      e.printStackTrace();
    }
    return ret;
  }
 
  /**
   * 释放锁
   * @param lockKey
   * @param value
   * @return
   */
  public boolean releaseLock(String lockKey, 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);
 
    Object result = redisTemplate.execute(redisScript,new StringRedisSerializer(),new StringRedisSerializer(), Collections.singletonList(lockKey),value);
    if(SUCCESS.equals(result)) {
      return true;
    }
 
    return false;
  }

   以上代码已经在我的项目中确切可以使用了。但是在使用的过程中遇到了许多问题。

   1:java.lang.IllegalStateException

   在返回值方面,会经常报IllegalStateException。

RedisScript<String> redisScript = new DefaultRedisScript<>(script, String.class);

   用String类型时候,经常会报类型转换异常。我在代码中使用的Long类型接收该类型,在命令行中我们也看到命令行结果返回的是数字0或者1,保险起见我们也可以用Object对象来接收结果集。

 2:ERR value is not an integer or out of range

  这个问题纠结了我一个下午至少,Redis报的异常都是很深的,从跟踪源码的时候看到,我们在调用redisTemplate.execute的方法时候,如果不传序列化的参数的时候,代码默认调用的是 Jdkserializationredisserializer 来进行序列化和反序列化操作,这是jdk自带的序列化操作,使用该序列化的对象必须要实现Serializable接口。所以该序列化接口是用于对实体类的序列化。

   所以在进行 execute 操作的时候,我们传入 Stringredisserializer,该序列化接口是专用于对字符串类型的序列化操作。具体的区别可以去这两个类的源码中看下他们的加密方式。 

因为时间以及个人能力的问题,对部分源码有点未理解,所以没有做到全方位的解读这些异常的原因,以后有机会会将源码细读并分析其异常原因。

到此这篇关于详解RedisTemplate下Redis分布式锁引发的系列问题的文章就介绍到这了,更多相关RedisTemplate 分布式锁内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Redis 相关文章推荐
解决redis sentinel 频繁主备切换的问题
Apr 12 Redis
Redis Cluster 字段模糊匹配及删除
May 27 Redis
详解Redis复制原理
Jun 04 Redis
了解Redis常见应用场景
Jun 23 Redis
redis中lua脚本使用教程
Nov 01 Redis
Redis+Lua脚本实现计数器接口防刷功能(升级版)
Feb 12 Redis
Redis调用Lua脚本及使用场景快速掌握
Mar 16 Redis
源码分析Redis中 set 和 sorted set 的使用方法
Mar 22 Redis
解决 Redis 秒杀超卖场景的高并发
Apr 12 Redis
Grafana可视化监控系统结合SpringBoot使用
Apr 19 Redis
Redis基本数据类型哈希Hash常用操作命令
Jun 01 Redis
如何使用注解方式实现 Redis 分布式锁
Jul 23 Redis
详解Redis实现限流的三种方式
Apr 27 #Redis
在K8s上部署Redis集群的方法步骤
Redis持久化与主从复制的实践
浅谈Redis在直播场景的实践方案
Apr 27 #Redis
redis限流的实际应用
Apr 24 #Redis
Redis安装启动及常见数据类型
redis配置文件中常用配置详解
Apr 14 #Redis
You might like
PHP 杂谈《重构-改善既有代码的设计》之四 简化条件表达式
2012/04/09 PHP
laravel 多图上传及图片的存储例子
2019/10/14 PHP
js中将具有数字属性名的对象转换为数组
2011/03/06 Javascript
javascript 运算数的求值顺序
2011/08/23 Javascript
PHP PDO操作总结
2014/11/17 Javascript
jQuery表单美化插件jqTransform使用详解
2015/04/12 Javascript
极易被忽视的javascript面试题七问七答
2016/02/15 Javascript
js判断空对象的实例(超简单)
2016/07/26 Javascript
利用JS判断鼠标移入元素的方向
2016/12/11 Javascript
jquery仿京东商品放大浏览页面
2017/06/06 jQuery
JS实现不用中间变量temp 实现两个变量值得交换方法
2018/02/04 Javascript
vue.js 使用axios实现下载功能的示例
2018/03/05 Javascript
vue下拉列表功能实例代码
2018/04/08 Javascript
js变量声明var使用与不使用的区别详解
2019/01/21 Javascript
[50:01]Ti4 冒泡赛第二天 NEWBEE vs Titan
2014/07/15 DOTA
在Python中操作列表之List.append()方法的使用
2015/05/20 Python
python递归全排列实现方法
2018/08/18 Python
对python requests发送json格式数据的实例详解
2018/12/19 Python
python学生管理系统
2019/01/30 Python
TensorFlow学习之分布式的TensorFlow运行环境
2020/02/05 Python
Python大批量搜索引擎图像爬虫工具详解
2020/11/16 Python
pytorch 计算Parameter和FLOP的操作
2021/03/04 Python
Data URI scheme详解和使用实例及图片base64编码实现方法
2014/05/08 HTML / CSS
湖南卫视在线视频媒体平台:芒果TV
2019/10/30 全球购物
俄罗斯童装网上商店:BebaKids
2020/06/06 全球购物
进程的查看和调度分别使用什么命令
2015/03/25 面试题
运动会广播稿400字
2014/01/25 职场文书
西门豹教学反思
2014/02/04 职场文书
倡议书范文
2014/04/16 职场文书
程序员求职信
2014/04/16 职场文书
公司仓管员岗位职责
2015/04/01 职场文书
兴趣班停课通知
2015/04/24 职场文书
《西门豹》教学反思
2016/02/23 职场文书
职场新人刚入职工作总结该怎么写?
2019/05/15 职场文书
php TP5框架生成二维码链接
2021/04/01 PHP
联想win10摄像头打不开怎么办?win10笔记本摄像头打不开解决办法
2022/04/08 数码科技