微信搜索《Java鱼仔》,每天一个知识点不错过
每天一个知识点
什么是接口的幂等性,如何实现接口幂等性?
(一)幂等性概念
幂等性原本是数学上的概念,用在接口上就可以理解为:同一个接口,多次发出同一个请求,必须保证操作只执行一次。
调用接口发生异常并且重复尝试时,总是会造成系统所无法承受的损失,所以必须阻止这种现象的发生。
比如下面这些情况,如果没有实现接口幂等性会有很严重的后果:
支付接口,重复支付会导致多次扣钱
订单接口,同一个订单可能会多次创建。
(二)幂等性的解决方案
唯一索引
使用唯一索引可以避免脏数据的添加,当插入重复数据时数据库会抛异常,保证了数据的唯一性。
乐观锁
这里的乐观锁指的是用乐观锁的原理去实现,为数据字段增加一个version字段,当数据需要更新时,先去数据库里获取此时的version版本号
1 | sql复制代码select version from tablename where xxx |
更新数据时首先和版本号作对比,如果不相等说明已经有其他的请求去更新数据了,提示更新失败。
1 | sql复制代码update tablename set count=count+1,version=version+1 where version=#{version} |
悲观锁
乐观锁可以实现的往往用悲观锁也能实现,在获取数据时进行加锁,当同时有多个重复请求时其他请求都无法进行操作
分布式锁
幂等的本质是分布式锁的问题,分布式锁正常可以通过redis或zookeeper实现;在分布式环境下,锁定全局唯一资源,使请求串行化,实际表现为互斥锁,防止重复,解决幂等。
token机制
token机制的核心思想是为每一次操作生成一个唯一性的凭证,也就是token。一个token在操作的每一个阶段只有一次执行权,一旦执行成功则保存执行结果。对重复的请求,返回同一个结果。token机制的应用十分广泛。
(三)token机制的实现
这里展示通过token机制实现接口幂等性的案例:github文末自取
首先引入需要的依赖:
1 | xml复制代码<dependency> |
3.1、配置请求的方法体和枚举类
首先配置一下通用的请求返回体
1 | java复制代码public class Response { |
以及返回code
1 | java复制代码public enum ResponseCode { |
3.2 自定义异常以及配置全局异常类
1 | java复制代码public class ServiceException extends RuntimeException{ |
配置全局异常捕获器
1 | java复制代码@ControllerAdvice |
3.3 编写创建Token和验证Token的接口以及实现类
1 | java复制代码@Service |
具体实现类,核心的业务逻辑都写在注释中了
1 | java复制代码@Service |
3.4 配置自定义注解
这是比较重要的一步,通过自定义注解在需要实现接口幂等性的方法上添加此注解,实现token验证
1 | java复制代码@Target({ElementType.METHOD}) |
接口拦截器
1 | java复制代码public class ApiIdempotentInterceptor implements HandlerInterceptor { |
3.5 配置拦截器以及redis
配置webConfig,添加拦截器
1 | java复制代码@Configuration |
配置redis,使得中文可以正常传输
1 | java复制代码@Configuration |
最后是controller
1 | java复制代码@RestController |
其余代码在文末github链接上自取
(四)结果验证
首先通过token接口创建一个token出来,此时redis中也存在了改token
在jmeter中同时运行50个请求,我们可以观察到,只有第一个请求校验成功,后续的请求均提示请勿重复操作。
jmeter压测文件(Token Plan.jmx)和代码自取:github自取
本文转载自: 掘金