面试官:你们系统是怎么实现分布式锁的?

:我们使用了redis的分布式锁。具体做法是后端接收到请求后加入一个分布式锁,如果加锁成功,就执行业务,如果加锁失败就等待锁或者拒绝请求。业务执行完成后释放锁。

面试官:能说一下具体使用的命令吗?

:我们使用的是SETNX命令,具体如下:

SETNX KEY_NAME VALUE

设置成功返回1,设置失败返回0。如下图,客户端1加锁成功,客户端2获取锁失败:

面试官:这样设置会不会有问题呢?如果加锁成功的客户端挂了怎么办?

:比如上图中的客户端1挂了,这个锁就不能释放了。可以设置一个过期时间,命令如下:

SET key value [EX seconds] [PX milliseconds] NX

面试官:设置了过期时间,如果业务还没有执行完成,但是redis锁过期了,怎么办?

:需要对锁进行续约。

面试官:能说一下具体怎么操作吗?

:设置锁成功后,启动一个watchdog,每隔一段时间(比如10s)为当前分布式锁续约,也就是每隔10s重新设置当前key的超时时间。命令如下:

EXPIRE <key> <seconds>

整个流程如下:

面试官:watchdog怎么实现呢?

:当客户端加锁成功后,可以启动一个定时任务,每隔10s(最好支持配置)来检测业务是否处理完成,检测的依据就是判断分布式锁的key是否还存在,如果存在,就进行续约。

面试官:如果当前线程已经处理完,这个key是被其他客户端写入的呢?

:可以为每个客户端指定一个clientID,在VALUE中增加一个clientID的前缀,这样在续锁的时候,可以判断当前分布式锁的value前缀来确定是不是当前客户端的,如果是再续锁,否则不做处理。

面试官:你们的续锁功能是自己实现的吗?

:我们用的redisson的分布式锁方案,使用redisson获取分布式锁非常简单,代码如下:

RLock lock = redisson.getLock("client-lock");
lock.lock();
try {//处理业务
} catch (Exception e) {//处理异常
} finally {lock.unlock();
}

具体原理是:如果客户端1加锁成功,这个分布式锁超时时间默认是30秒(可以通过Config.lockWatchdogTimeout来修改)。加锁成功后,就会启动一个watchdog,watchdog是一个后台线程,会每隔10秒检查一下客户端1是否还持有锁key,如果是,就延长锁key的生存时间,延长操作就是再次把锁key的超时时间设置成30s。

面试官:redisson里的定时器怎么实现的?

:redisson定时器使用的是netty-common包中的HashedWheelTime来实现的。

面试官:如果client1宕机了,这时分布式锁还可以续期吗?

:因为分布式锁的续期是在客户端执行的,所以如果client1宕机了,续期线程就不能工作了,也就不能续期了。这时应该把分布式锁删除,让其他客户端来获取。

面试官:那如果client1宕机了,其他客户端需要等待30s才能有机会获取到锁,有办法立刻删除锁吗?

:因为client1宕机了,只能等到超时时间后锁被自动删除。如果要立刻删除,需要增加额外的工作,比如增加哨兵机制,让哨兵来维护所有redis客户端的列表。哨兵定时监控客户端是否宕机,如果检测到宕机,立刻删除这个客户端的锁。如下图:

这里的哨兵并不是redis的哨兵,而且为了检测客户端故障业务系统自己做的哨兵。

面试官:如果不用redisson,怎么实现分布式锁续锁呢?比如springboot2.0默认使用redis客户端是Lettuce。

:Lettuce并没有提供像redisson这样的watchdog机制,所以续锁需要业务系统自己实现。可以分为以下几步来实现:

  1. 加锁的命令,我们参照spring包里的分布式锁代码,如果锁存在并且是当前客户端加的锁,那就续锁,如果锁不存在,则加锁。代码如下:

private static final String OBTAIN_LOCK_SCRIPT ="local lockClientId = redis.call('GET', KEYS[1])\n" +"if lockClientId == ARGV[1] then\n" +"  redis.call('PEXPIRE', KEYS[1], ARGV[2])\n" +"  return true\n" +"elseif not lockClientId then\n" +"  redis.call('SET', KEYS[1], ARGV[1], 'PX', ARGV[2])\n" +"  return true\n" +"end\n" +"return false";
  1. 把锁保存在一个数据结构里,比如HashMap,定时任务定时扫描这个map,对每个锁进行续锁操作。代码如下:

private final Map<String, RedisLock> locks = new ConcurrentHashMap<>();
  1. 续锁命令

private static final String RENEW_LOCK_SCRIPT ="local lockClientId = redis.call('GET', KEYS[1])\n" +"if lockClientId == ARGV[1] then\n" +"  redis.call('PEXPIRE', KEYS[1], ARGV[2])\n" +"  return true\n" +"end\n" +"return false";

如果锁是当前客户端加的,那就续锁,否则失败。

  1. 写一个定时任务,定时执行续锁代码:

redisTemplate.execute(renewLockScript,Collections.singletonList(lockKey), clientId,String.valueOf(expireAfter));

面试官:这个问题就聊到这里,咱们下一个问题...

阿里二面:redis分布式锁过期了但业务还没有执行完,怎么办相关推荐

  1. 还不知道 Redis 分布式锁的背后原理?还不赶快学习一下

    前言 以前在学校做小项目的时候,用到Redis,基本也只是用来当作缓存.可阿粉在工作中发现,Redis在生产中并不只是当作缓存这么简单.在阿粉接触到的项目中,Redis起到了一个分布式锁的作用,具体情 ...

  2. redis 分布式锁的 5个坑,真是又大又深

    引言 最近项目上线的频率颇高,连着几天加班熬夜,身体有点吃不消精神也有些萎靡,无奈业务方催的紧,工期就在眼前只能硬着头皮上了.脑子浑浑噩噩的时候,写的就不能叫代码,可以直接叫做Bug.我就熬夜写了一个 ...

  3. redis分布式锁的这些坑,我怀疑你是假的开发

    摘要:用锁遇到过哪些问题? 一.白话分布式 什么是分布式,用最简单的话来说,就是为了较低单个服务器的压力,将功能分布在不同的机器上面:就比如: 本来一个程序员可以完成一个项目:需求->设计-&g ...

  4. 详解Redis分布式锁

    最近首度应用"分布式锁",现在想想,分布式锁不是孤立的技能点,这其实就是跨主机的线程同步. 分布式锁是"线程同步"的延续 最近首度应用"分布式锁&qu ...

  5. Redis分布式锁抽丝剥茧

    之前码甲哥写了两篇有关线程安全的文章: •你管这叫线程安全?•.NET八股文:线程同步技术解读 分布式锁是"线程同步"的延续 最近首度应用"分布式锁",现在想想 ...

  6. 深入理解Redis分布式锁

    分布式环境下很多时候都需要使用到分布式锁,通常情况下 Redis 的实现方式是比较常见的. 文章目录 前言部分 什么是分布式锁 Redis分布式锁方案一:SETNX + EXPIRE Redis分布式 ...

  7. Redis 作者 Antirez 讲如何实现分布式锁?Redis 实现分布式锁天然的缺陷分析Redis分布式锁的正确使用姿势!...

    Redis分布式锁基本原理 采用 redis 实现分布式锁,主要是利用其单线程命令执行的特性,一般是 setnx, 只会有一个线程会执行成功,也就是只有一个线程能成功获取锁:看着很完美. 然而-- 看 ...

  8. redis分布式锁解决方案

    一.什么是分布式锁? 线程锁:主要用来给方法.代码块加锁.当某个方法或代码使用锁,在同一时刻仅有一个线程执行该方法或该代码段.线程锁只在同一JVM中有效果,因为线程锁的实现在根本上是依靠线程之间共享内 ...

  9. Redis分布式锁一文全攻略

    分布式锁概念 分布式锁其实就是,控制分布式系统的不同进程共同访问共享资源的一种锁的实现.如果不同系统或同一个系统的不同主机去访问一个共享的临界资源,往往需要互斥来防止彼此干扰,以保证一致性. 分布式锁 ...

最新文章

  1. Python+OpenCV实现自动扫雷,创造属于自己的世界记录!
  2. Struts2依赖的JAR包
  3. vue --- 购物车页面
  4. 网段和子网的区别_电焊石笼网与普通石笼网区别
  5. linux mysql搭建禅道详细教程_linux安装禅道的步骤
  6. 蚂蚁集团上市造富:员工激励达1376.9亿元,人均超800万!
  7. maven打包生成source.jar
  8. python对于字典d d.get(x、y)_给定字典 d ,哪个选项对 d.get(x, y) 的描述是正确的?_学小易找答案...
  9. xdebug断点调试原理
  10. 图论算法及其matlab实现_BLDC有感FOC算法理论及其STM32软硬件实现
  11. 写给数据小白:怎么让你的分析结论超出预期,不再是废纸一堆
  12. mysql将一个表的字段更新到另一个表中
  13. matlab irandon函数,在路上●我的年青●悠忽两年
  14. Pidgin for windows 与MSN、ICQ、QQ、YAHOO、GoogleTalk、AIM/AOL等网络聊天工具互联互通的新型聊天软件
  15. postgresql表复制
  16. 编译时出现stripped of unavailable superclass
  17. 神经网络理解:前向传播与反向传播
  18. 气动调节阀与电动调节阀的区别
  19. 沙发后面墙挂什么画 你需要一幅像样的装饰画
  20. python怎么将字符串逆序_python中如何把一个字符串顺序逆序反转的几种方法?

热门文章

  1. python open方法下file模块_python-linecache模块读取文件用法
  2. python自带intertool模块找不到_Python itertools模块:生成迭代器(示例分析)
  3. mysql 触发器 赋值_MYSQL的触发器中 变量赋值
  4. 简单探讨JavaScript 与 TypeScript之间的联系
  5. VBS遍历Excel工作表的方法
  6. 单片机一个月能入门么?单片机工程师能干到多少岁?
  7. linux deploy ENV 目录,手机安装linux deploy 安装和配置
  8. php rfc3986规范,「PSR 规范」PSR-7 HTTP 消息接口规范
  9. php递归内存,PHP递归的三种常用方式
  10. P2057 [SHOI2007]善意的投票 (最大流最小割)