详解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实现限流的三种方式
Apr 27 Redis
Redis IP地址的绑定的实现
May 08 Redis
浅谈Redis中的RDB快照
Jun 29 Redis
在项目中使用redis做缓存的一些思路
Sep 14 Redis
聊聊redis-dump工具安装问题
Jan 18 Redis
redis击穿 雪崩 穿透超详细解决方案梳理
Mar 17 Redis
Redis如何使用乐观锁(CAS)保证数据一致性
Mar 25 Redis
Redis官方可视化工具RedisInsight安装使用教程
Apr 19 Redis
Redis 报错 error:NOAUTH Authentication required
May 15 Redis
Redis入门基础常用操作命令整理
Jun 01 Redis
Redis全局ID生成器的实现
Jun 05 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
无线电广播与收音机发展的历史回眸
2021/03/02 无线电
Cygwin中安装PHP方法步骤
2015/07/04 PHP
JavaScript 封装Ajax传递的数据代码
2009/06/05 Javascript
一个简单的javascript类定义例子
2009/09/12 Javascript
用document.documentElement取代document.body的原因分析
2009/11/12 Javascript
键盘KeyCode值列表汇总
2013/11/26 Javascript
JQuery与JS里submit()的区别示例介绍
2014/02/17 Javascript
jQuery实现多按钮单击变色
2014/11/27 Javascript
JavaScript使用DeviceOne开发实战(三)仿微信应用
2015/12/02 Javascript
json格式的javascript对象用法分析
2016/07/04 Javascript
Bootstrap分页插件之Bootstrap Paginator实例详解
2016/10/15 Javascript
读Javascript高性能编程重点笔记
2016/12/21 Javascript
Node.js 使用流实现读写同步边读边写功能
2017/09/11 Javascript
小程序ios音频播放没声音问题的解决
2018/07/11 Javascript
vue 点击按钮实现动态挂载子组件的方法
2018/09/07 Javascript
jquery登录的异步验证操作示例
2019/05/09 jQuery
使用layui日期控件laydate对开始和结束时间进行联动控制的方法
2019/09/06 Javascript
layui监听下拉选框选中值变化的方法(包含监听普通下拉选框)
2019/09/24 Javascript
python写的ARP攻击代码实例
2014/06/04 Python
Python实现从百度API获取天气的方法
2015/03/11 Python
python传递参数方式小结
2015/04/17 Python
在Django的form中使用CSS进行设计的方法
2015/07/18 Python
在 Python 应用中使用 MongoDB的方法
2017/01/05 Python
对Python定时任务的启动和停止方法详解
2019/02/19 Python
500行Python代码打造刷脸考勤系统
2019/06/03 Python
Python基础学习之类与实例基本用法与注意事项详解
2019/06/17 Python
Python爬虫爬取ts碎片视频+验证码登录功能
2021/02/22 Python
HTML5拖拽文件到浏览器并实现文件上传下载功能代码
2013/06/06 HTML / CSS
Lou & Grey美国官网:主打舒适性面料服饰
2017/12/21 全球购物
Nike德国官网:Nike.com (DE)
2018/11/13 全球购物
在c#中using和new这两个关键字有什么意义
2013/05/19 面试题
教师求职信范文
2014/05/24 职场文书
房产协议书范本
2014/10/18 职场文书
2014年辅导员工作总结
2014/11/18 职场文书
学生通报表扬范文
2015/05/04 职场文书
Pytest中conftest.py的用法
2021/06/27 Python