什么是接口的幂等性,如何实现接口幂等性?一文搞定

本文涉及的产品
云数据库 Tair(兼容Redis),内存型 2GB
Redis 开源版,标准版 2GB
推荐场景:
搭建游戏排行榜
简介: 幂等性原本是数学上的概念,用在接口上就可以理解为:同一个接口,多次发出同一个请求,必须保证操作只执行一次。 调用接口发生异常并且重复尝试时,总是会造成系统所无法承受的损失,所以必须阻止这种现象的发生。 比如下面这些情况,如果没有实现接口幂等性会有很严重的后果: 支付接口,重复支付会导致多次扣钱 订单接口,同一个订单可能会多次创建。

微信搜索《Java鱼仔》,每天一个知识点不错过


每天一个知识点


什么是接口的幂等性,如何实现接口幂等性?


(一)幂等性概念


幂等性原本是数学上的概念,用在接口上就可以理解为:同一个接口,多次发出同一个请求,必须保证操作只执行一次。 调用接口发生异常并且重复尝试时,总是会造成系统所无法承受的损失,所以必须阻止这种现象的发生。 比如下面这些情况,如果没有实现接口幂等性会有很严重的后果: 支付接口,重复支付会导致多次扣钱 订单接口,同一个订单可能会多次创建。


(二)幂等性的解决方案


唯一索引使用唯一索引可以避免脏数据的添加,当插入重复数据时数据库会抛异常,保证了数据的唯一性。


乐观锁这里的乐观锁指的是用乐观锁的原理去实现,为数据字段增加一个version字段,当数据需要更新时,先去数据库里获取此时的version版本号


select version from tablename where xxx

更新数据时首先和版本号作对比,如果不相等说明已经有其他的请求去更新数据了,提示更新失败。

update tablename setcount=count+1,version=version+1where version=#{version}

悲观锁乐观锁可以实现的往往用悲观锁也能实现,在获取数据时进行加锁,当同时有多个重复请求时其他请求都无法进行操作


分布式锁幂等的本质是分布式锁的问题,分布式锁正常可以通过redis或zookeeper实现;在分布式环境下,锁定全局唯一资源,使请求串行化,实际表现为互斥锁,防止重复,解决幂等。


token机制token机制的核心思想是为每一次操作生成一个唯一性的凭证,也就是token。一个token在操作的每一个阶段只有一次执行权,一旦执行成功则保存执行结果。对重复的请求,返回同一个结果。token机制的应用十分广泛。


(三)token机制的实现


这里展示通过token机制实现接口幂等性的案例:github文末自取 首先引入需要的依赖:


<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-redis</artifactId></dependency><dependency><groupId>org.apache.commons</groupId><artifactId>commons-lang3</artifactId><version>3.4</version></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId></dependency>

3.1、配置请求的方法体和枚举类


首先配置一下通用的请求返回体

publicclassResponse {
privateintstatus;
privateStringmsg;
privateObjectdata;
//省略get、set、toString、无参有参构造方法}

以及返回code

publicenumResponseCode {
// 通用模块 1xxxxILLEGAL_ARGUMENT(10000, "参数不合法"),
REPETITIVE_OPERATION(10001, "请勿重复操作"),
    ;
ResponseCode(Integercode, Stringmsg) {
this.code=code;
this.msg=msg;
    }
privateIntegercode;
privateStringmsg;
publicIntegergetCode() {
returncode;
    }
publicvoidsetCode(Integercode) {
this.code=code;
    }
publicStringgetMsg() {
returnmsg;
    }
publicvoidsetMsg(Stringmsg) {
this.msg=msg;
    }
}

3.2 自定义异常以及配置全局异常类

publicclassServiceExceptionextendsRuntimeException{
privateStringcode;
privateStringmsg;
//省略get、set、toString以及构造方法}

配置全局异常捕获器

@ControllerAdvicepublicclassMyControllerAdvice {
@ResponseBody@ExceptionHandler(ServiceException.class)
publicResponseserviceExceptionHandler(ServiceExceptionexception){
Responseresponse=newResponse(Integer.valueOf(exception.getCode()),exception.getMsg(),null);
returnresponse;
    }
}

3.3 编写创建Token和验证Token的接口以及实现类

@ServicepublicinterfaceTokenService {
publicResponsecreateToken();
publicResponsecheckToken(HttpServletRequestrequest);
}

具体实现类,核心的业务逻辑都写在注释中了

@ServicepublicclassTokenServiceImplimplementsTokenService {
@AutowiredprivateRedisTemplateredisTemplate;
@OverridepublicResponsecreateToken() {
//生成uuid当作tokenStringtoken=UUID.randomUUID().toString().replaceAll("-","");
//将生成的token存入redis中redisTemplate.opsForValue().set(token,token);
//返回正确的结果信息Responseresponse=newResponse(0,token.toString(),null);
returnresponse;
    }
@OverridepublicResponsecheckToken(HttpServletRequestrequest) {
//从请求头中获取tokenStringtoken=request.getHeader("token");
if (StringUtils.isBlank(token)){
//如果请求头token为空就从参数中获取token=request.getParameter("token");
//如果都为空抛出参数异常的错误if (StringUtils.isBlank(token)){
thrownewServiceException(ResponseCode.ILLEGAL_ARGUMENT.getCode().toString(),ResponseCode.ILLEGAL_ARGUMENT.getMsg());
            }
        }
//如果redis中不包含该token,说明token已经被删除了,抛出请求重复异常if (!redisTemplate.hasKey(token)){
thrownewServiceException(ResponseCode.REPETITIVE_OPERATION.getCode().toString(),ResponseCode.REPETITIVE_OPERATION.getMsg());
        }
//删除tokenBooleandel=redisTemplate.delete(token);
//如果删除不成功(已经被其他请求删除),抛出请求重复异常if (!del){
thrownewServiceException(ResponseCode.REPETITIVE_OPERATION.getCode().toString(),ResponseCode.REPETITIVE_OPERATION.getMsg());
        }
returnnewResponse(0,"校验成功",null);
    }
}

3.4 配置自定义注解


这是比较重要的一步,通过自定义注解在需要实现接口幂等性的方法上添加此注解,实现token验证

@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
public@interfaceApiIdempotent {
}

接口拦截器

publicclassApiIdempotentInterceptorimplementsHandlerInterceptor {
@AutowiredprivateTokenServicetokenService;
@OverridepublicbooleanpreHandle(HttpServletRequestrequest, HttpServletResponseresponse, Objecthandler) throwsException {
if (!(handlerinstanceofHandlerMethod)) {
returntrue;
        }
HandlerMethodhandlerMethod= (HandlerMethod) handler;
Methodmethod=handlerMethod.getMethod();
ApiIdempotentmethodAnnotation=method.getAnnotation(ApiIdempotent.class);
if (methodAnnotation!=null){
// 校验通过放行,校验不通过全局异常捕获后输出返回结果tokenService.checkToken(request);
        }
returntrue;
    }
@OverridepublicvoidpostHandle(HttpServletRequestrequest, HttpServletResponseresponse, Objecthandler, ModelAndViewmodelAndView) throwsException {
    }
@OverridepublicvoidafterCompletion(HttpServletRequestrequest, HttpServletResponseresponse, Objecthandler, Exceptionex) throwsException {
    }
}

3.5 配置拦截器以及redis


配置webConfig,添加拦截器

@ConfigurationpublicclassWebConfigimplementsWebMvcConfigurer {
@OverridepublicvoidaddInterceptors(InterceptorRegistryregistry) {
registry.addInterceptor(apiIdempotentInterceptor());
    }
@BeanpublicApiIdempotentInterceptorapiIdempotentInterceptor() {
returnnewApiIdempotentInterceptor();
    }
}

配置redis,使得中文可以正常传输

@ConfigurationpublicclassRedisConfig {
//自定义的redistemplate@Bean(name="redisTemplate")
publicRedisTemplate<String,Object>redisTemplate(RedisConnectionFactoryfactory){
//创建一个RedisTemplate对象,为了方便返回key为string,value为ObjectRedisTemplate<String,Object>template=newRedisTemplate<>();
template.setConnectionFactory(factory);
//设置json序列化配置Jackson2JsonRedisSerializerjackson2JsonRedisSerializer=newJackson2JsonRedisSerializer(Object.class);
ObjectMapperobjectMapper=newObjectMapper();
objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
objectMapper.activateDefaultTyping(LaissezFaireSubTypeValidator.instance);
//string的序列化StringRedisSerializerstringRedisSerializer=newStringRedisSerializer();
//key采用string的序列化方式template.setKeySerializer(stringRedisSerializer);
//value采用jackson的序列化方式template.setValueSerializer(jackson2JsonRedisSerializer);
//hashkey采用string的序列化方式template.setHashKeySerializer(stringRedisSerializer);
//hashvalue采用jackson的序列化方式template.setHashValueSerializer(jackson2JsonRedisSerializer);
template.afterPropertiesSet();
returntemplate;
    }
}

最后是controller

@RestController@RequestMapping("/token")
publicclassTokenController {
@AutowiredprivateTokenServicetokenService;
@GetMappingpublicResponsetoken(){
returntokenService.createToken();
    }
@PostMapping("checktoken")
publicResponsechecktoken(HttpServletRequestrequest){
returntokenService.checkToken(request);
    }
}

其余代码在文末github链接上自取


(四)结果验证


首先通过token接口创建一个token出来,此时redis中也存在了改token


网络异常,图片无法展示
|


在jmeter中同时运行50个请求,我们可以观察到,只有第一个请求校验成功,后续的请求均提示请勿重复操作。


网络异常,图片无法展示
|


网络异常,图片无法展示
|


jmeter压测文件(Token Plan.jmx)和代码自取:github自取



相关实践学习
基于Redis实现在线游戏积分排行榜
本场景将介绍如何基于Redis数据库实现在线游戏中的游戏玩家积分排行榜功能。
云数据库 Redis 版使用教程
云数据库Redis版是兼容Redis协议标准的、提供持久化的内存数据库服务,基于高可靠双机热备架构及可无缝扩展的集群架构,满足高读写性能场景及容量需弹性变配的业务需求。 产品详情:https://www.aliyun.com/product/kvstore &nbsp; &nbsp; ------------------------------------------------------------------------- 阿里云数据库体验:数据库上云实战 开发者云会免费提供一台带自建MySQL的源数据库&nbsp;ECS 实例和一台目标数据库&nbsp;RDS实例。跟着指引,您可以一步步实现将ECS自建数据库迁移到目标数据库RDS。 点击下方链接,领取免费ECS&amp;RDS资源,30分钟完成数据库上云实战!https://developer.aliyun.com/adc/scenario/51eefbd1894e42f6bb9acacadd3f9121?spm=a2c6h.13788135.J_3257954370.9.4ba85f24utseFl
相关文章
|
5月前
接口幂等性设计
接口幂等性设计
54 1
|
5月前
|
NoSQL 关系型数据库 MySQL
接口防刷 && 接口幂等性问题
接口防刷 && 接口幂等性问题
70 0
|
前端开发 NoSQL JavaScript
常见接口和服务幂等性问题及解决方案
常见接口和服务幂等性问题及解决方案
384 0
|
5月前
|
存储 缓存 安全
接口的幂等性
接口的幂等性
62 0
|
SQL 缓存 NoSQL
接口的幂等性设计和防重保证,详细分析幂等性的几种实现方法
本篇文章详细说明了幂等性,解释了什么是幂等性,幂等性的使用场景,讨论了幂等和防重的概念。分析了幂等性的情况以及如何设计幂等性服务。阐述了幂等性实现防重的几种策略,包括乐关锁,防重表,分布式锁,token令牌以及支付缓冲区。
6306 0
接口的幂等性设计和防重保证,详细分析幂等性的几种实现方法
|
2月前
|
缓存 NoSQL Java
接口幂等该如何设计和实现
本文探讨了程序开发中常见的重复操作问题,如多次点击生成多余订单或支付、RPC调用失败后的重试机制滥用及非法重复请求等。通过接口幂等性设计可有效解决这类问题,确保相同请求多次执行结果一致无副作用。文章详细解释了幂等性的概念及其重要性,并提供了具体的设计与实现方法,包括使用唯一标识符、设计幂等操作、事务处理及缓存策略。此外,还讨论了实现幂等性接口所带来的好处,如并发请求处理、失败请求管理及系统集成等,并提出了验证接口幂等性的策略。通过这些技术和方法的应用,可以显著提升系统的稳定性和用户体验。
|
2月前
|
SQL 索引
分布式之接口幂等性
分布式之接口幂等性
37 2
|
4月前
|
数据库 API 网络架构
浅谈应用接口的幂等性
【6月更文挑战第2天】本文介绍幂等性是计算和网络通信中的重要概念,确保同一操作执行多次不会改变结果。在数据库操作中,查询、删除(同一数据)和特定更新是幂等的,而插入和累加更新不是。幂等性和安全性(如GET、HEAD等方法)确保多次请求无副作用,对涉及金钱的操作尤为重要。
48 0
|
5月前
|
API
什么是接口幂等
什么是接口幂等
130 0
|
5月前
|
存储 缓存 数据库
接口幂等有哪些实现方式
接口幂等有哪些实现方式
39 0