分布式锁实战

背景

柯苏远写于2024年4月7日 20点10分

公司的主要业务线是国外电商,之前是在客户下订单,以及厂商对客户订单进行打包发货等关键节点都会给客户发送对应邮件。

可是由于邮件没有短信那么及时,方便,为了提高客户的体验和下单率准备在整个系统接入短信。

所以这里就涉及到要和国外通讯商对接短信发送的接口。

这里主要是两个接口:一个是获取token接口,一个是发送短信接口。获取token的接口一个小时之内只能获取5次,一次获取到的token有效期是一个小时。

简单 token 缓存设计

根据背景描述,显而易见的是,token必须要缓存起来,因为一个小时就只能获取5次,所以必须尽量重复使用,以下是我的设想的接口调用情况:

从直观感受上来讲,确实是可以了,我们通常的设计一般情况下都不会考虑多线程并发的情况。

但是!!! 我们的业务项目是电商欸,肯定会存在并发的可能。

所以我将方案给我老大看了之后,我老大一眼就看出我没有考虑并发问题,要我重新去设计。

并发 token 缓存设计

由于上一版的核心就是没考虑并发,如果此时100个线程同时去缓存获取并且同时没获取到,再然后同时去调用了短信的api,那么这个api一分钟5次的限制一下子就到了峰值。

考虑到并发点,我引入了分布式锁,对之前方案做了更新,如下图所示:

整个流程图看起来复杂了很多,其实和上面方案比较起来主要就两点:

  1. 在调用api之前加了分布式锁,在取得token之后,释放分布式锁。
  2. 如果在一个线程持有分布式锁的情况,其它线程进来没获取锁,就让它们冷处理,类似一个while循环操作,然后睡眠200毫秒,循环15次去缓存里取。如果15次都没有从缓存里取到token对象,那么就要进行去抢夺分布式锁,如果抢夺成功那么执行调用api流程,如果失败,那么再去缓存看看,有的话直接返回token对象,没有的话再去抢夺分布式锁。

具体实现 & 当前问题

具体实现分布式锁:

  • 在项目里用的是redisTemplate,有一个setIfAbsent,然后再结合expire(项目里是10s),用这两个命令来获取分布式锁。
  • 释放的时候直接就是delete。

其实这种实现方式是不严谨的,有如下缺点:

  • setIfAbsent和expire:这两个不是原子命令,如果setIfAbsent成功,由于某种原因expire没有成功,那么就会造成死锁,这个分布式锁会得不到释放。
  • expire 这里设置的过期时间是10s,但是如果我业务执行时间大于10s呢?分布式锁就失效了,那还搞个毛线?其它线程也进我的临界区了,线程不安全了。
  • delete是直接删除的,没有考虑某种未知错误造成删除失败的原因。这个其实是小问题,即使删除失败也有超时的保证。

这三个目前存在的问题和老大聊了,老大说确实是有这问题,然后后面研究一下redisson,然后完善一下这个分布式锁,后面等手头活少点去研究一下。

总结

  1. api token 缓存设计

  2. 多线程下 api token 缓存设计

  3. 分布式锁的实现 & 目前存在的问题

  4. 后续目标:研究redisson实现分布式锁。

相关推荐
Tech Synapse44 分钟前
Java根据前端返回的字段名进行查询数据的方法
java·开发语言·后端
.生产的驴1 小时前
SpringCloud OpenFeign用户转发在请求头中添加用户信息 微服务内部调用
spring boot·后端·spring·spring cloud·微服务·架构
微信-since811921 小时前
[ruby on rails] 安装docker
后端·docker·ruby on rails
代码吐槽菌3 小时前
基于SSM的毕业论文管理系统【附源码】
java·开发语言·数据库·后端·ssm
豌豆花下猫3 小时前
Python 潮流周刊#78:async/await 是糟糕的设计(摘要)
后端·python·ai
YMWM_3 小时前
第一章 Go语言简介
开发语言·后端·golang
码蜂窝编程官方3 小时前
【含开题报告+文档+PPT+源码】基于SpringBoot+Vue的虎鲸旅游攻略网的设计与实现
java·vue.js·spring boot·后端·spring·旅游
hummhumm4 小时前
第 25 章 - Golang 项目结构
java·开发语言·前端·后端·python·elasticsearch·golang
J老熊4 小时前
JavaFX:简介、使用场景、常见问题及对比其他框架分析
java·开发语言·后端·面试·系统架构·软件工程
AuroraI'ncoding4 小时前
时间请求参数、响应
java·后端·spring