zoukankan      html  css  js  c++  java
  • 保证接口幂等性的解决方案(后台)

    假如有个服务提供一个接口(服务部署在多个服务机器),接着有个接口是付款接口。用户在前端上操作的时候,一个订单不小心发起了两次支付请求,然后这两个请求分散在了这个服务部署的不同的机器上,结果一个订单扣款扣两次。这样的场景,就是接口没有保证幂等性的结果。

    保证幂等性的核心

    1.对于每个请求必须有一个唯一的标识。

    2.每次处理完请求之后,必须有一个记录标识这个请求处理过了。

    3.每次接收请求需要进行判断之前是否处理过的逻辑处理。

    常见解决方案

    1.业务表内唯一索引

    如果要对创建销售出库单的接口保证幂等性,也就是说人家网络超时,重复调用的时候,保证一个订单只能有一个对应的销售出库单。针对销售出库单的表的订单id,创建一个唯一索引,你如果接口被重试,同一个订单创建一个销售出库单的话,一定会违反唯一索引,那么此时会报错。

    2.业务表内状态机

    修改订单状态,比如说将订单状态修改为【待发货】的时候,订单的状态其实就变为了【待发货】。

    update order set status = '待发货' where status = '待付款' and id = 1;

    这时候如果id为1的订单接口被重复调用了,即使再执行一次这个操作也不会有效果,因为这时候该订单记录的状态字段已经改变了,SQL并不会命中该记录。

    在这种业务场景中也是通常都会有逻辑判断的,比如当前是否处于某个状态,然后才能流转到下一个状态(状态为待付款的才能流转到待发货)。

    3.基于版本号的更新

    id name age version
    1 yanggb 18 1

    给业务表内添加一个版本号的字段,如果要调用一个接口去更新年龄之前,就需要先查一下他的版本号是多少,然后调用接口的时候带上版本号。

    在接口里保证分布式接口的幂等性(在更新的SQL中添加version的条件判断):

    update user set age = 21, version = version + 1 where id = 1 and version = 1;

    这样,多次提交的请求,因为版本号(version)都一样,因为第一次请求执行成功之后version已经+1了,则后面的请求因为version对应不上,都不会被执行。

    4.基于MySQL的去重表 / 基于Redis的去重

    比如说接口方法为changeAge(1, 21, 1),可以将所有的参数拼接成一个字符串,或者是从这些入参里选择一些参数(可以唯一标识这一次请求的一些参数),每次请求进来,在操作之前先校验这个字符串,如果校验通过则继续执行操作,校验失败则跳过。

    如果基于MySQL,可以单独搞一个表出来(可以就一个字段),在这个表上的一个字段建一个唯一索引,插入的记录值就是前面拼接的字符串。因为第一次请求到达的时候,这个字符串在表中还不存在对应的记录,则会往表中插入该记录,并继续执行业务逻辑。后面的请求再到达的时候,因为这个字符串在表中已经存在了对应的记录了,唯一索引就会报一个冲突出来,这次插入就会失败,后续的业务逻辑也就会跳过执行了。

    如果接口调用量很大,并发很高,还可以选择使用Redis。同样是拼接一个字符串串出来,直接set设置到Redis里去,如果下一次请求再过来,会发现这个key已经存在了,那么这个时候就不能执行了,因为已经可以知道出现了重复调用的情况了。

    幂等的缺点

    幂等是为了简化客户端逻辑处理,但是却增加了服务提供者的逻辑和成本,是否有必要在系统中引入幂等,需要根据具体场景具体分析,因此除了业务上的特殊要求外,一般建议是尽量不提供幂等的接口。

    1.增加了额外控制幂等的业务逻辑,复杂化了业务功能。

    2.把并行执行的功能改为串行执行,降低了执行效率。

    "人不能太深情得活着,也不能总是追缅过去。"

  • 相关阅读:
    XJ20夏令营做题记录(长期更新)
    洛谷P6623——[省选联考 2020 A 卷] 树
    [游记] 2020ZJOI 爆零记
    CF1017G——The Tree
    CF715E—— Complete the Permutations
    学习笔记——树的初步整理
    学习笔记——DP初步整理
    洛谷P5290——春节十二响
    POJ3017——Cut the Sequence(单调队列+堆优化DP)
    Java控制整形输入的法子
  • 原文地址:https://www.cnblogs.com/yanggb/p/11713309.html
Copyright © 2011-2022 走看看