处理接口幂等性的两种常见方案

在上周发布的 TienChin 项目视频中,我和大家一共梳理了六种幂等性解决方案接口幂等性处理算是一个非常常见的需求了,我们在很多项目中其实都会遇到。今天我们来看看两种比较简单的实现思路

1. 接口幂等性实现方案梳理

其实接口幂等性的实现方案还是蛮多的,我这里和小伙伴们分享两种比较常见的方案。

1.1 基于 Token

基于 Token 这种方案的实现思路很简单,整个流程分两步:

  • 客户端发送请求,从服务端获取一个 Token 令牌,每次请求获取到的都是一个全新的令牌。
  • 客户端发送请求的时候,携带上第一步的令牌,处理请求之前,先校验令牌是否存在,当请求处理成功,就把令牌删除掉。
  • 大致的思路就是上面这样,当然具体的实现则会复杂很多,有很多细节需要注意,松哥之前也专门录过这种方案的视频,小伙伴们可以参考下,录了两个视频,一个是基于拦截器处理的,还有一个是基于 AOP 切面处理的:

    基于拦截器处理(视频一):

    基于 AOP 切面处理(视频二):

    1.2 基于请求参数校验

    最近在 TienChin 项目中使用的是另外一种方案,这种方案是基于请求参数来判断的,如果在短时间内,同一个接口接收到的请求参数相同,那么就认为这是重复的请求,拒绝处理,大致上就是这么个思路。

    相比于第一种方案,第二种方案相对来说省事一些,因为只有一次请求,不需要专门去服务端拿令牌。在高并发环境下这种方案优势比较明显。

    所以今天我就来和大家聊聊第二种方案的实现,后面在 TienChin 项目视频中也会和大家细讲。

    2. 基于请求参数的校验

    首先我们新建一个 Spring Boot 项目,引入 Web 和 Redis 依赖,新建完成后,先来配置一下 Redis 的基本信息,如下:

    spring.redis.host=localhostspring.redis.port=6379spring.redis.password=123

    为了后续 Redis 操作方便,我们再来对 Redis 进行一个简单封装,如下:

    @Componentpublic class RedisCache { @Autowired public RedisTemplate redisTemplate; public void setCacheObject(final String key, final T value, final Integer timeout, final TimeUnit timeUnit) { redisTemplate.opsForValue().set(key, value, timeout, timeUnit); } public T getCacheObject(final String key) { ValueOperations operation = redisTemplate.opsForValue(); return operation.get(key); }}

    这个比较简单,一个存数据,一个读数据。

    接下来我们自定义一个注解,在需要进行幂等性处理的接口上,添加该注解即可,将来这个接口就会自动的进行幂等性处理。

    @Inherited@Target(ElementType.METHOD)@Retention(RetentionPolicy.RUNTIME)@Documentedpublic @interface RepeatSubmit { /** * 间隔时间(ms),小于此时间视为重复提交 */ public int interval() default 5000; /** * 提示消息 */ public String message() default “不允许重复提交,请稍候再试”;}

    这个注解我们通过拦截器来进行解析,解析代码如下:

    public abstract class RepeatSubmitInterceptor implements HandlerInterceptor { @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { if (handler instanceof HandlerMethod) { HandlerMethod handlerMethod = (HandlerMethod) handler; Method method = handlerMethod.getMethod(); RepeatSubmit annotation = method.getAnnotation(RepeatSubmit.class); if (annotation != null) { if (this.isRepeatSubmit(request, annotation)) { Map map = new HashMap(); map.put(“status”, 500); map.put(“msg”, annotation.message()); response.setContentType(“application/json;charset=utf-8”); response.getWriter().write(new ObjectMapper().writeValueAsString(map)); return false; } } return true; } else { return true; } } /** * 验证是否重复提交由子类实现具体的防重复提交的规则 * * @param request * @return * @throws Exception */ public abstract boolean isRepeatSubmit(HttpServletRequest request, RepeatSubmit annotation);}

    这个拦截器是一个抽象类,将接口方法拦截下来,然后找到接口上的 @RepeatSubmit 注解,调用 isRepeatSubmit 方法去判断是否是重复提交的数据,该方法在这里是一个抽象方法,我们需要再定义一个类继承自这个抽象类,在新的子类中,可以有不同的幂等性判断逻辑,这里我们就是根据 URL 地址+参数 来判断幂等性条件是否满足:

    @Componentpublic class SameUrlDataInterceptor extends RepeatSubmitInterceptor { public final String REPEAT_PARAMS = “repeatParams”; public final String REPEAT_TIME = “repeatTime”; public final static String REPEAT_SUBMIT_KEY = “REPEAT_SUBMIT_KEY”; private String header = “Authorization”; @Autowired private RedisCache redisCache; @SuppressWarnings(“unchecked”) @Override public boolean isRepeatSubmit(HttpServletRequest request, RepeatSubmit annotation) { String nowParams = “”; if (request instanceof RepeatedlyRequestWrapper) { RepeatedlyRequestWrapper repeatedlyRequest = (RepeatedlyRequestWrapper) request; try { nowParams = repeatedlyRequest.getReader().readLine(); } catch (IOException e) { e.printStackTrace(); } } // body参数为空,获取Parameter的数据 if (StringUtils.isEmpty(nowParams)) { try { nowParams = new ObjectMapper().writeValueAsString(request.getParameterMap()); } catch (JsonProcessingException e) { e.printStackTrace(); } } Map nowDataMap = new HashMap(); nowDataMap.put(REPEAT_PARAMS, nowParams); nowDataMap.put(REPEAT_TIME, System.currentTimeMillis()); // 请求地址(作为存放cache的key值) String url = request.getRequestURI(); // 唯一值(没有消息头则使用请求地址) String submitKey = request.getHeader(header); // 唯一标识(指定key + url + 消息头) String cacheRepeatKey = REPEAT_SUBMIT_KEY + url + submitKey; Object sessionObj = redisCache.getCacheObject(cacheRepeatKey); if (sessionObj != null) { Map sessionMap = (Map) sessionObj; if (compareParams(nowDataMap, sessionMap) && compareTime(nowDataMap, sessionMap, annotation.interval())) { return true; } } redisCache.setCacheObject(cacheRepeatKey, nowDataMap, annotation.interval(), TimeUnit.MILLISECONDS); return false; } /** * 判断参数是否相同 */ private boolean compareParams(Map nowMap, Map preMap) { String nowParams = (String) nowMap.get(REPEAT_PARAMS); String preParams = (String) preMap.get(REPEAT_PARAMS); return nowParams.equals(preParams); } /** * 判断两次间隔时间 */ private boolean compareTime(Map nowMap, Map preMap, int interval) { long time1 = (Long) nowMap.get(REPEAT_TIME); long time2 = (Long) preMap.get(REPEAT_TIME); if ((time1 – time2) < interval) { return true; } return false; }}

    我们来看下具体的实现逻辑:

  • 首先判断当前的请求对象是不是 RepeatedlyRequestWrapper,如果是,说明当前的请求参数是 JSON,那么就通过 IO 流将参数读取出来,这块小伙伴们要结合上篇文章共同来理解,否则可能会觉得云里雾里的,传送门JSON 数据读一次就没了,怎么办?。
  • 如果在第一步中,并没有拿到参数,那么说明参数可能并不是 JSON 格式,而是 key-value 格式,那么就以 key-value 的方式读取出来参数,并将之转为一个 JSON 字符串。
  • 接下来构造一个 Map,将前面读取到的参数和当前时间存入到 Map 中。
  • 接下来构造存到 Redis 中的数据的 key,这个 key 由固定前缀 + 请求 URL 地址 + 请求头的认证令牌组成,这块请求头的令牌还是非常重要需要有的,只有这样才能区分出来当前用户提交的数据(如果是 RESTful 风格的接口,那么为了区分,也可以将接口的请求方法作为参数拼接到 key 中)。
  • 接下来就去 Redis 中获取数据,获取到之后,分别去比较参数是否相同以及时间是否过期。
  • 如果判断都没问题,返回 true,表示这个请求重复了。
  • 否则返回说明这是用户对这个接口第一次提交数据或者是已经过了时间窗口了,那么就把参数字符串重新缓存到 Redis 中,并返回 false,表示请求没问题。
  • 好啦,做完这一切,最后我们再来配置一下拦截器即可:

    @Configurationpublic class WebConfig implements WebMvcConfigurer { @Autowired RepeatSubmitInterceptor repeatSubmitInterceptor; @Override public void addInterceptors(InterceptorRegistry registry) { registry.addInterceptor(repeatSubmitInterceptor) .addPathPatterns(“/**”); }}

    如此,我们的接口幂等性就处理好啦 在需要的时候,就可以直接在接口上使用啦:

    @RestControllerpublic class HelloController { @PostMapping(“/hello”) @RepeatSubmit(interval = 100000) public String hello(@RequestBody String msg) { System.out.println(“msg = ” + msg); return “hello”; }}

    好啦,公众号后台回复 RepeatSubmit 可以下载本文源码哦。

    郑重声明:本文内容及图片均整理自互联网,不代表本站立场,版权归原作者所有,如有侵权请联系管理员(admin#wlmqw.com)删除。
    (0)
    用户投稿
    上一篇 2022年6月21日
    下一篇 2022年6月21日

    相关推荐

    • 桑启确认重做,普攻加血移除,明世隐大砍一刀,大幅度降低双抗

      大家好,我是阿呆。这里将会给你带来王者荣耀最新的爆料内容。 随着王者荣耀体验服更新之后,官方又在体验服中,重大调整了两位英雄,分别是明世隐和桑启,这两位英雄一个是当前版本的强大辅助…

      2022年8月5日
    • 养胃误区,当心“养出”胃病

      您一定知道怎么养胃,而且随口就能说出一个所以然来,但是民间流传着的各种养胃知识真的正确吗?真的什么情况下都适宜吗?但是如果听信了错误的方法,不仅不能达到养胃的目的,反而会加重胃的负…

      2022年8月17日
    • 海康威视:在水利防汛领域提供河湖管护智能管理平台、水灾害防御指挥调度平台等

      【海康威视:在水利防汛领域提供河湖管护智能管理平台、水灾害防御指挥调度平台等】财联社6月24日电,海康威视在互动平台表示,海康威视在水利防汛领域提供河湖管护智能管理平台、智慧水安综…

      2022年6月25日
    • 12个月!从失业到2亿粉丝

      据雅虎财经网站报道,日前,世界最大的加密货币交易所币安(Binance)宣布,聘请生活在意大利的塞内加尔籍创作者卡比·莱姆(Khaby Lame)成为其全球品牌大使。这位在TikT…

      2022年7月20日
    • 三笔2换1交易方案:76人为哈登上双保险,火箭押宝申京老大哥

      作为今夏的大赢家,76人队在休赛期完成了一系列的人员调整与补强,在ESPN等多家媒体公布的休赛期操作评分中均获得A评分。 球队先是用丹尼格林换回了梅尔顿,接着又低价续约了哈登,这一…

      2022年8月24日
    • 萤石北斗星视频锁DL30V:超清可视对讲通话

      萤石智能家居 众所周知,萤石作为具备安防基因的企业,在视频技术方面沉淀多年,视频能力一直处于行业领先地位。基于萤石擅长的视频能力,其所生产的视频锁的品质也是相当硬核,兼具便捷与安全…

      2022年6月24日
    • IT设备行业借助CRM数字化转型,重塑企业新增长

      “疫情时代”企业传统线下营销模式屡屡受挫,以往得心应手的市场或内部管理手段,在这个特殊的时期,举步维艰。越来越多的企业尝试开辟新的路径,实现企业在内外部之间的交流。 随着互联网人口…

      2022年6月16日
    • 正确洗头的10个方法,你get到了吗?

      头发经常油,是美丽的烦恼。经常食用辣、油加上压力大的工作,会影响人的内部系统,加剧头皮问题。平时我们应该如何洗好头?为您推荐几个小方法。 1、用手搓起小泡后再接触头皮。 洗发的时候…

      2022年8月15日
    • 胡凌:超越企业视角,互联互通的意义在于建立统一数据大市场

      8月12日,由北京科技创新中心研究基地主办的 “互联网治理中的开放与封闭研讨会”在线上线下同步举行,来自北京多所高校、实务部门、业内专家学者围绕平台互联互通的理论基础与实践难题,互…

      2022年8月29日
    • 现在互联网到底有哪些赚钱的项目?

      作者:I’m零微课链接:https://www.zhihu.com/question/302128426/answer/637965131来源:知乎著作权归作者所有。商…

      2022年6月19日

    联系我们

    联系邮箱:admin#wlmqw.com
    工作时间:周一至周五,10:30-18:30,节假日休息