0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

一個(gè)注解,優(yōu)雅的實(shí)現(xiàn)接口冪等性!

jf_ro2CN3Fa ? 來源:芋道源碼 ? 2023-08-26 14:36 ? 次閱讀


一、什么是冪等性?

簡(jiǎn)單來說,就是對(duì)一個(gè)接口執(zhí)行重復(fù)的多次請(qǐng)求,與一次請(qǐng)求所產(chǎn)生的結(jié)果是相同的,聽起來非常容易理解,但要真正的在系統(tǒng)中要始終保持這個(gè)目標(biāo),是需要很嚴(yán)謹(jǐn)?shù)脑O(shè)計(jì)的,在實(shí)際的生產(chǎn)環(huán)境下,我們應(yīng)該保證任何接口都是冪等的,而如何正確的實(shí)現(xiàn)冪等,就是本文要討論的內(nèi)容。

基于 Spring Boot + MyBatis Plus + Vue & Element 實(shí)現(xiàn)的后臺(tái)管理系統(tǒng) + 用戶小程序,支持 RBAC 動(dòng)態(tài)權(quán)限、多租戶、數(shù)據(jù)權(quán)限、工作流、三方登錄、支付、短信、商城等功能

  • 項(xiàng)目地址:https://github.com/YunaiV/ruoyi-vue-pro
  • 視頻教程:https://doc.iocoder.cn/video/

二、哪些請(qǐng)求天生就是冪等的?

首先,我們要知道查詢類的請(qǐng)求一般都是天然冪等的,除此之外,刪除請(qǐng)求在大多數(shù)情況下也是冪等的,但是ABA場(chǎng)景下除外。

舉一個(gè)簡(jiǎn)單的例子

比如,先請(qǐng)求了一次刪除A的操作,但由于響應(yīng)超時(shí),又自動(dòng)請(qǐng)求了一次刪除A的操作,如果在兩次請(qǐng)求之間,又插入了一次A,而實(shí)際上新插入的這一次A,是不應(yīng)該被刪除的,這就是ABA問題,不過,在大多數(shù)業(yè)務(wù)場(chǎng)景中,ABA問題都是可以忽略的。

除了查詢和刪除之外,還有更新操作,同樣的更新操作在大多數(shù)場(chǎng)景下也是天然冪等的,其例外是也會(huì)存在ABA的問題,更重要的是,比如執(zhí)行update table set a = a + 1 where v = 1這樣的更新就非冪等了。

最后,就還剩插入了,插入大多數(shù)情況下都是非冪等的,除非是利用數(shù)據(jù)庫(kù)唯一索引來保證數(shù)據(jù)不會(huì)重復(fù)產(chǎn)生。

基于 Spring Cloud Alibaba + Gateway + Nacos + RocketMQ + Vue & Element 實(shí)現(xiàn)的后臺(tái)管理系統(tǒng) + 用戶小程序,支持 RBAC 動(dòng)態(tài)權(quán)限、多租戶、數(shù)據(jù)權(quán)限、工作流、三方登錄、支付、短信、商城等功能

  • 項(xiàng)目地址:https://github.com/YunaiV/yudao-cloud
  • 視頻教程:https://doc.iocoder.cn/video/

三、為什么需要冪等

1.超時(shí)重試

當(dāng)發(fā)起一次RPC請(qǐng)求時(shí),難免會(huì)因?yàn)?a href="http://wenjunhu.com/v/tag/1722/" target="_blank">網(wǎng)絡(luò)不穩(wěn)定而導(dǎo)致請(qǐng)求失敗,一般遇到這樣的問題我們希望能夠重新請(qǐng)求一次,正常情況下沒有問題,但有時(shí)請(qǐng)求實(shí)際上已經(jīng)發(fā)出去了,只是在請(qǐng)求響應(yīng)時(shí)網(wǎng)絡(luò)異常或者超時(shí),此時(shí),請(qǐng)求方如果再重新發(fā)起一次請(qǐng)求,那被請(qǐng)求方就需要保證冪等了。

2.異步回調(diào)

異步回調(diào)是提升系統(tǒng)接口吞吐量的一種常用方式,很明顯,此類接口一定是需要保證冪等性的。

3.消息隊(duì)列

現(xiàn)在常用的消息隊(duì)列框架,比如:Kafka、RocketMQ、RabbitMQ在消息傳遞時(shí)都會(huì)采取At least once原則(也就是至少一次原則,在消息傳遞時(shí),不允許丟消息,但是允許有重復(fù)的消息),既然消息隊(duì)列不保證不會(huì)出現(xiàn)重復(fù)的消息,那消費(fèi)者自然要保證處理邏輯的冪等性了。

四、實(shí)現(xiàn)冪等的關(guān)鍵因素

關(guān)鍵因素1

冪等唯一標(biāo)識(shí),可以叫它冪等號(hào)或者冪等令牌或者全局ID,總之就是客戶端與服務(wù)端一次請(qǐng)求時(shí)的唯一標(biāo)識(shí),一般情況下由客戶端來生成,也可以讓第三方來統(tǒng)一分配。

關(guān)鍵因素2

有了唯一標(biāo)識(shí)以后,服務(wù)端只需要確保這個(gè)唯一標(biāo)識(shí)只被使用一次即可,一種常見的方式就是利用數(shù)據(jù)庫(kù)的唯一索引。

五、注解實(shí)現(xiàn)冪等性

下面演示一種利用Redis來實(shí)現(xiàn)的方式。

1.自定義注解

importjava.lang.annotation.ElementType;
importjava.lang.annotation.Retention;
importjava.lang.annotation.RetentionPolicy;
importjava.lang.annotation.Target;

@Target(value=ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public@interfaceIdempotent{

/**
*參數(shù)名,表示將從哪個(gè)參數(shù)中獲取屬性值。
*獲取到的屬性值將作為KEY。
*
*@return
*/
Stringname()default"";

/**
*屬性,表示將獲取哪個(gè)屬性的值。
*
*@return
*/
Stringfield()default"";

/**
*參數(shù)類型
*
*@return
*/
Classtype();

}

2.統(tǒng)一的請(qǐng)求入?yún)?duì)象

@Data
publicclassRequestData<T>{

privateHeaderheader;

privateTbody;

}


@Data
publicclassHeader{

privateStringtoken;

}

@Data
publicclassOrder{

StringorderNo;

}

3.AOP處理

importcom.springboot.micrometer.annotation.Idempotent;
importcom.springboot.micrometer.entity.RequestData;
importcom.springboot.micrometer.idempotent.RedisIdempotentStorage;
importorg.aspectj.lang.ProceedingJoinPoint;
importorg.aspectj.lang.annotation.Around;
importorg.aspectj.lang.annotation.Aspect;
importorg.aspectj.lang.annotation.Pointcut;
importorg.aspectj.lang.reflect.MethodSignature;
importorg.springframework.stereotype.Component;

importjavax.annotation.Resource;
importjava.lang.reflect.Method;
importjava.util.Map;

@Aspect
@Component
publicclassIdempotentAspect{

@Resource
privateRedisIdempotentStorageredisIdempotentStorage;

@Pointcut("@annotation(com.springboot.micrometer.annotation.Idempotent)")
publicvoididempotent(){
}

@Around("idempotent()")
publicObjectmethodAround(ProceedingJoinPointjoinPoint)throwsThrowable{
MethodSignaturesignature=(MethodSignature)joinPoint.getSignature();
Methodmethod=signature.getMethod();
Idempotentidempotent=method.getAnnotation(Idempotent.class);

Stringfield=idempotent.field();
Stringname=idempotent.name();
ClassclazzType=idempotent.type();

Stringtoken="";

Objectobject=clazzType.newInstance();
MapparamValue=AopUtils.getParamValue(joinPoint);
if(objectinstanceofRequestData){
RequestDataidempotentEntity=(RequestData)paramValue.get(name);
token=String.valueOf(AopUtils.getFieldValue(idempotentEntity.getHeader(),field));
}

if(redisIdempotentStorage.delete(token)){
returnjoinPoint.proceed();
}
return"重復(fù)請(qǐng)求";
}
}
importorg.aspectj.lang.ProceedingJoinPoint;
importorg.aspectj.lang.reflect.CodeSignature;

importjava.lang.reflect.Field;
importjava.util.HashMap;
importjava.util.Map;

publicclassAopUtils{

publicstaticObjectgetFieldValue(Objectobj,Stringname)throwsException{
Field[]fields=obj.getClass().getDeclaredFields();
Objectobject=null;
for(Fieldfield:fields){
field.setAccessible(true);
if(field.getName().toUpperCase().equals(name.toUpperCase())){
object=field.get(obj);
break;
}
}
returnobject;
}


publicstaticMapgetParamValue(ProceedingJoinPointjoinPoint){
Object[]paramValues=joinPoint.getArgs();
String[]paramNames=((CodeSignature)joinPoint.getSignature()).getParameterNames();
Mapparam=newHashMap<>(paramNames.length);

for(inti=0;ireturnparam;
}
}

4.Token值生成

importcom.springboot.micrometer.idempotent.RedisIdempotentStorage;
importcom.springboot.micrometer.util.IdGeneratorUtil;
importorg.springframework.web.bind.annotation.RequestMapping;
importorg.springframework.web.bind.annotation.RestController;

importjavax.annotation.Resource;

@RestController
@RequestMapping("/idGenerator")
publicclassIdGeneratorController{

@Resource
privateRedisIdempotentStorageredisIdempotentStorage;

@RequestMapping("/getIdGeneratorToken")
publicStringgetIdGeneratorToken(){
StringgenerateId=IdGeneratorUtil.generateId();
redisIdempotentStorage.save(generateId);
returngenerateId;
}

}
publicinterfaceIdempotentStorage{

voidsave(StringidempotentId);

booleandelete(StringidempotentId);
}
importorg.springframework.data.redis.core.RedisTemplate;
importorg.springframework.stereotype.Component;

importjavax.annotation.Resource;
importjava.io.Serializable;
importjava.util.concurrent.TimeUnit;

@Component
publicclassRedisIdempotentStorageimplementsIdempotentStorage{

@Resource
privateRedisTemplateredisTemplate;

@Override
publicvoidsave(StringidempotentId){
redisTemplate.opsForValue().set(idempotentId,idempotentId,10,TimeUnit.MINUTES);
}

@Override
publicbooleandelete(StringidempotentId){
returnredisTemplate.delete(idempotentId);
}
}
importjava.util.UUID;

publicclassIdGeneratorUtil{

publicstaticStringgenerateId(){
returnUUID.randomUUID().toString();
}

}

5. 請(qǐng)求示例

調(diào)用接口之前,先申請(qǐng)一個(gè)token,然后帶著服務(wù)端返回的token值,再去請(qǐng)求。

importcom.springboot.micrometer.annotation.Idempotent;
importcom.springboot.micrometer.entity.Order;
importcom.springboot.micrometer.entity.RequestData;
importorg.springframework.web.bind.annotation.RequestBody;
importorg.springframework.web.bind.annotation.RequestMapping;
importorg.springframework.web.bind.annotation.RestController;

@RestController
@RequestMapping("/order")
publicclassOrderController{

@RequestMapping("/saveOrder")
@Idempotent(name="requestData",type=RequestData.class,field="token")
publicStringsaveOrder(@RequestBodyRequestDatarequestData){
return"success";
}

}

請(qǐng)求獲取token值。

533d7310-43cc-11ee-a2ef-92fbcf53809c.png

帶著token值,第一次請(qǐng)求成功。

53540ddc-43cc-11ee-a2ef-92fbcf53809c.png

第二次請(qǐng)求失敗。

53676bde-43cc-11ee-a2ef-92fbcf53809c.png


聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 接口
    +關(guān)注

    關(guān)注

    33

    文章

    8615

    瀏覽量

    151307
  • RPC
    RPC
    +關(guān)注

    關(guān)注

    0

    文章

    111

    瀏覽量

    11540
  • 管理系統(tǒng)
    +關(guān)注

    關(guān)注

    1

    文章

    2521

    瀏覽量

    35946

原文標(biāo)題:一個(gè)注解,優(yōu)雅的實(shí)現(xiàn)接口冪等性!

文章出處:【微信號(hào):芋道源碼,微信公眾號(hào):芋道源碼】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    離線計(jì)算中的和DataWorks中的相關(guān)事項(xiàng)

    多次相同的消息,針對(duì)同筆交易的付款也不應(yīng)該在重試過程中扣多次錢。曾見過個(gè)案例,有個(gè)對(duì)于個(gè)
    發(fā)表于 02-27 13:24

    在高并發(fā)下怎么保證接口

    前言 接口性問題,對(duì)于開發(fā)人員來說,是個(gè)跟語(yǔ)言無關(guān)的公共問題。本文分享了些解決這類問題非
    的頭像 發(fā)表于 05-14 10:23 ?1819次閱讀
    在高并發(fā)下怎么保證<b class='flag-5'>接口</b>的<b class='flag-5'>冪</b><b class='flag-5'>等</b><b class='flag-5'>性</b>?

    注解定義Bean及開發(fā)

    注解本質(zhì)是個(gè)繼承了Annotation 的特殊接口,其具體實(shí)現(xiàn)類是Java 運(yùn)行時(shí)生成的動(dòng)態(tài)代理類。
    發(fā)表于 08-02 10:26 ?446次閱讀

    什么是?關(guān)于接口的解決方案

    這里的樂觀鎖指的是用樂觀鎖的原理去實(shí)現(xiàn),為數(shù)據(jù)字段增加個(gè)version字段,當(dāng)數(shù)據(jù)需要更新時(shí),先去數(shù)據(jù)庫(kù)里獲取此時(shí)的version版本號(hào)
    發(fā)表于 10-09 10:19 ?1955次閱讀

    分析解決)的方法

    這個(gè)概念,是個(gè)數(shù)學(xué)上的概念,即:f……(f(f(x))) = f(x)。用在計(jì)算機(jī)領(lǐng)域,指的是系統(tǒng)里的接口或方法對(duì)外的
    的頭像 發(fā)表于 10-14 10:08 ?969次閱讀

    Spring Boot實(shí)現(xiàn)接口的4種方案

    個(gè)數(shù)學(xué)與計(jì)算機(jī)學(xué)概念,在數(shù)學(xué)中某元運(yùn)算為
    的頭像 發(fā)表于 11-08 10:21 ?1007次閱讀

    如何設(shè)計(jì)個(gè)優(yōu)雅的API接口

    種是API接口提供方給出AK/SK兩個(gè)值,雙方約定用SK作為簽名中的密鑰。AK接口調(diào)用方作為header中的accessKey傳遞給API接口
    的頭像 發(fā)表于 12-20 14:23 ?1637次閱讀

    什么是?實(shí)現(xiàn)原理

    在編程中個(gè)操作的特點(diǎn)是其任意多次執(zhí)行所產(chǎn)生的影響均與次執(zhí)行的影響相同。
    發(fā)表于 01-05 10:40 ?6151次閱讀

    給定個(gè)接口,要用戶自定義動(dòng)態(tài)實(shí)現(xiàn)并上傳熱部署

    考慮到用戶實(shí)現(xiàn)接口的兩種方式,使用spring上下文管理的方式,或者不依賴spring管理的方式,這里稱它們?yōu)?b class='flag-5'>注解方式和反射方式。calculate方法對(duì)應(yīng)注解方式,add方法對(duì)應(yīng)反射
    的頭像 發(fā)表于 01-06 14:14 ?567次閱讀

    如何實(shí)現(xiàn)個(gè)注解進(jìn)行數(shù)據(jù)脫敏

    、測(cè)試 后記 ? 本文主要分享什么是數(shù)據(jù)脫敏,如何優(yōu)雅的在項(xiàng)目中運(yùn)用個(gè)注解實(shí)現(xiàn)數(shù)據(jù)脫敏,為項(xiàng)目進(jìn)行賦能。希望能給你們帶來幫助。 什么是數(shù)據(jù)
    的頭像 發(fā)表于 06-14 09:37 ?1025次閱讀
    如何<b class='flag-5'>實(shí)現(xiàn)</b><b class='flag-5'>一</b><b class='flag-5'>個(gè)</b><b class='flag-5'>注解</b>進(jìn)行數(shù)據(jù)脫敏

    基于接口解決方案

    接口是指無論調(diào)用接口的次數(shù)是次還是多次,對(duì)于同
    的頭像 發(fā)表于 09-30 16:27 ?440次閱讀
    基于<b class='flag-5'>接口</b><b class='flag-5'>冪</b><b class='flag-5'>等</b><b class='flag-5'>性</b>解決方案

    和非請(qǐng)求的些定義和分析

    最近在做項(xiàng)目的過程中,有個(gè)需求是在客戶端 HTTP 請(qǐng)求失敗后,增加個(gè)重試機(jī)制,然后我就翻了些有關(guān)“重試”的庫(kù),找到
    的頭像 發(fā)表于 10-17 10:50 ?827次閱讀

    接口統(tǒng)異常優(yōu)雅處理介紹及實(shí)戰(zhàn)

    Spring在3.2版本增加了個(gè)注解@ControllerAdvice,可以與@ExceptionHandler、@InitBinder、@ModelAttribute
    的頭像 發(fā)表于 10-22 16:01 ?765次閱讀
    <b class='flag-5'>接口</b>統(tǒng)<b class='flag-5'>一</b>異常<b class='flag-5'>優(yōu)雅</b>處理介紹及實(shí)戰(zhàn)

    為什么要實(shí)現(xiàn)校驗(yàn) 如何實(shí)現(xiàn)接口校驗(yàn)

    前端重復(fù)提交表單:在填寫些表格時(shí)候,用戶填寫完成提交,很多時(shí)候會(huì)因網(wǎng)絡(luò)波動(dòng)沒有及時(shí)對(duì)用戶做出提交成功響應(yīng),致使用戶認(rèn)為沒有成功提交,然后直點(diǎn)提交按鈕,這時(shí)就會(huì)發(fā)生重復(fù)提交表單請(qǐng)求。
    的頭像 發(fā)表于 02-20 14:14 ?1231次閱讀

    探索LabVIEW編程接口原理與實(shí)踐

    原來是數(shù)學(xué)上的概念,在編程領(lǐng)域可以理解為:多次請(qǐng)求某一個(gè)資源或執(zhí)行某一個(gè)操作時(shí)應(yīng)該具有唯一性
    的頭像 發(fā)表于 02-29 10:24 ?637次閱讀
    探索LabVIEW編程<b class='flag-5'>接口</b><b class='flag-5'>冪</b><b class='flag-5'>等</b><b class='flag-5'>性</b>原理與實(shí)踐