java接口防重提交的處理方法
1.什么是接口防重?
在一定的時(shí)間內(nèi)多次請(qǐng)求同一接口,同一參數(shù)。由于請(qǐng)求是健康請(qǐng)求,會(huì)執(zhí)行正常的業(yè)務(wù)邏輯,從而產(chǎn)生大量的廢數(shù)據(jù)。
2.問(wèn)題的產(chǎn)生及引發(fā)的問(wèn)題
舉一個(gè)最簡(jiǎn)單的例子:日常開(kāi)發(fā)中crud在業(yè)務(wù)系統(tǒng)中普遍存在,在服務(wù)端沒(méi)有做任何處理,客戶(hù)端沒(méi)有做節(jié)流、防抖等限流操作時(shí),同一秒一個(gè)用戶(hù)點(diǎn)了兩次新增按鈕,導(dǎo)致數(shù)據(jù)庫(kù)中存在同樣兩條數(shù)據(jù),其結(jié)果可想而知,同理修改、刪除同樣的道理;查詢(xún)本身具有冪等性,但是在同一秒鐘同樣的操作,查詢(xún)多次和一次,有區(qū)別嗎?區(qū)別大了去了,不談?dòng)脩?hù)體驗(yàn)如何,光是網(wǎng)絡(luò)開(kāi)銷(xiāo)、流量占用、帶給服務(wù)器的壓力等等,生產(chǎn)中一點(diǎn)小的問(wèn)題,如何不及時(shí)處理,可能會(huì)引發(fā)災(zāi)難性bug。
3.處理方法
- 第一種:前臺(tái)在請(qǐng)求接口的時(shí)候,傳遞一個(gè)唯一值,然后在對(duì)應(yīng)接口判斷該唯一值,在一定的時(shí)間內(nèi)是否被消費(fèi)過(guò)
- 第二種:采用Spring AOP理念,實(shí)現(xiàn)請(qǐng)求的切割,在請(qǐng)求執(zhí)行到某個(gè)方法或某層時(shí)候,開(kāi)始攔截進(jìn)行,獲取該請(qǐng)求的參數(shù),用戶(hù)信息,請(qǐng)求地址,存入redis中并放置過(guò)期時(shí)間,進(jìn)行防重(推薦使用)
4.談?wù)勔韵聝煞N處理方法的利弊
- 第一種:局限性太高,前臺(tái)必須傳遞一個(gè)唯一值,就算請(qǐng)求到達(dá)指定后臺(tái)服務(wù),寫(xiě)一個(gè)攔截器,需要配置太多不需要攔截的方法,也許你會(huì)說(shuō),可以攔截有規(guī)則的請(qǐng)求地址,這樣真的好嗎?
- 第二種(推薦):采用AOP面向切面編程的思想,在不污染源代碼的情況下,進(jìn)行增強(qiáng)功能,切入到要防重的接口上,實(shí)現(xiàn)統(tǒng)一防重處理、業(yè)務(wù)解耦。此處采用AOP + 自定義注解,靈活實(shí)現(xiàn)防重功能。
5.具體代碼(采用第二種)
注解類(lèi)
import java.lang.annotation.*; /** ?* 防重 ?* @date 2020/8/12 ?* @return ?*/ //標(biāo)識(shí)該注解用于方法上 @Target({ElementType.METHOD}) //申明該注解為運(yùn)行時(shí)注解,編譯后改注解不會(huì)被遺棄 @Retention(RetentionPolicy.RUNTIME) //javadoc工具記錄 @Documented public @interface PreventSubmit? { }
切面類(lèi)
import com.qianxian.common.exception.AppException; import com.qianxian.common.util.TokenUtil; import lombok.extern.slf4j.Slf4j; import org.apache.commons.lang3.StringUtils; import org.aspectj.lang.ProceedingJoinPoint; import org.aspectj.lang.annotation.Around; import org.aspectj.lang.annotation.Aspect; import org.redisson.api.RLock; import org.redisson.api.RedissonClient; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.data.redis.core.StringRedisTemplate; import org.springframework.stereotype.Component; import org.springframework.web.context.request.RequestAttributes; import org.springframework.web.context.request.RequestContextHolder; import javax.servlet.http.HttpServletRequest; import java.util.Arrays; import java.util.concurrent.TimeUnit; /** ?* 防重復(fù)提交 ?* @date 2020/8/12 ?* @return ?*/ @Component @Aspect @Slf4j public class PreventSubmitAspect { ? ? /** ? ? ?* 放重redis前綴 ? ? ?*/ ? ? private static String API_PREVENT_SUBMIT = "api:preventSubmit:"; ? ? /** ? ? ?* 放重分布式鎖前綴 ? ? ?*/ ? ? private static String API_LOCK_PREVENT_SUBMIT = "api:preventSubmit:lock:"; ? ? /** ? ? ?* 失效時(shí)間 ? ? ?*/ ? ? private static Integer INVALID_NUMBER = 3; ? ? /** ? ? ?* redis ? ? ?*/ ? ? @Autowired ? ? private StringRedisTemplate stringRedisTemplate; ? ? /** ? ? ?* 分布式鎖 ? ? ?*/ ? ? @Autowired ? ? private RedissonClient redissonClient; ? ? /** ? ? ?* 防重 ? ? ?* @date 2020/8/12 ? ? ?* @return ? ? ?*/ ? ? @Around("@annotation(com.qianxian.user.annotation.PreventSubmit)") ? ? public Object preventSubmitAspect(ProceedingJoinPoint joinPoint) throws Throwable { ? ? ? ? RLock lock = null; ? ? ? ? try { ? ? ? ? ? ? //獲取目標(biāo)方法的參數(shù) ? ? ? ? ? ? Object[] args = joinPoint.getArgs(); ? ? ? ? ? ? //獲取當(dāng)前request請(qǐng)求 ? ? ? ? ? ? RequestAttributes requestAttributes = RequestContextHolder.getRequestAttributes(); ? ? ? ? ? ? HttpServletRequest request = (HttpServletRequest) requestAttributes.resolveReference(RequestAttributes.REFERENCE_REQUEST); ? ? ? ? ? ? //獲取請(qǐng)求地址 ? ? ? ? ? ? String requestUri = request.getRequestURI(); ? ? ? ? ? ? //獲取用戶(hù)ID ? ? ? ? ? ? Long userId = null; ? ? ? ? ? ? try { ? ? ? ? ? ? ? ? userId = TokenUtil.getUserId(request); ? ? ? ? ? ? }catch (Exception e){} ? ? ? ? ? ? //拼接鎖前綴,采用同一方法,同一用戶(hù),同一接口 ? ? ? ? ? ? String temp = requestUri.concat(Arrays.asList(args).toString()) + (userId != null ? userId : ""); ? ? ? ? ? ? temp = temp.replaceAll("/",""); ? ? ? ? ? ? //拼接rediskey ? ? ? ? ? ? String lockPrefix = API_LOCK_PREVENT_SUBMIT.concat(temp); ? ? ? ? ? ? String redisPrefix = API_PREVENT_SUBMIT.concat(temp); ? ? ? ? ? ? /** ? ? ? ? ? ? ?* 對(duì)同一方法同一用戶(hù)同一參數(shù)加鎖,即使獲取不到用戶(hù)ID,每個(gè)用戶(hù)請(qǐng)求數(shù)據(jù)也會(huì)不一致,不會(huì)造成接口堵塞 ? ? ? ? ? ? ?*/ ? ? ? ? ? ? lock = this.redissonClient.getLock(lockPrefix); ? ? ? ? ? ? lock.lock(); ? ? ? ? ? ? String flag = this.stringRedisTemplate.opsForValue().get(redisPrefix); ? ? ? ? ? ? if(StringUtils.isNotEmpty(flag)){ ? ? ? ? ? ? ? ? throw new AppException("您當(dāng)前的操作太頻繁了,請(qǐng)稍后再試!"); ? ? ? ? ? ? } ? ? ? ? ? ? //存入redis,設(shè)置失效時(shí)間 ? ? ? ? ? ? this.stringRedisTemplate.opsForValue() ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?.set(redisPrefix,redisPrefix,INVALID_NUMBER, TimeUnit.SECONDS); ? ? ? ? ? ? //執(zhí)行目標(biāo)方法 ? ? ? ? ? ? Object result = joinPoint.proceed(args); ? ? ? ? ? ? return result; ? ? ? ? }finally { ? ? ? ? ? ? if(lock != null){ ? ? ? ? ? ? ? ? lock.unlock(); ? ? ? ? ? ? } ? ? ? ? } ? ? } }
到此這篇關(guān)于java接口防重提交的處理方法的文章就介紹到這了,更多相關(guān)java接口防重提交內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java/Spring項(xiàng)目的包開(kāi)頭為什么是com詳解
這篇文章主要介紹了Java/Spring項(xiàng)目的包開(kāi)頭為什么是com的相關(guān)資料,在Java中包命名遵循域名反轉(zhuǎn)規(guī)則,即使用公司的域名反轉(zhuǎn)作為包的前綴,以確保其全球唯一性和避免命名沖突,這種規(guī)則有助于邏輯分層、代碼可讀性提升和標(biāo)識(shí)代碼來(lái)源,需要的朋友可以參考下2024-10-10MybatisPlus自定義Sql實(shí)現(xiàn)多表查詢(xún)的示例
這篇文章主要介紹了MybatisPlus自定義Sql實(shí)現(xiàn)多表查詢(xún)的示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08在Springboot中Mybatis與Mybatis-plus的區(qū)別詳解
MyBatis是一個(gè)優(yōu)秀的持久層框架,它對(duì)JDBC的操作數(shù)據(jù)庫(kù)的過(guò)程進(jìn)行封裝,MyBatisPlus (簡(jiǎn)稱(chēng) MP)是一個(gè) MyBatis的增強(qiáng)工具,在 MyBatis 的基礎(chǔ)上只做增強(qiáng)不做改變,為簡(jiǎn)化開(kāi)發(fā)、提高效率而生,本文將給大家介紹了在Springboot中Mybatis與Mybatis-plus的區(qū)別2023-12-12IDEA 將 SpringBoot 項(xiàng)目打包成jar的方法
這篇文章主要介紹了IDEA 將 SpringBoot 項(xiàng)目打包成jar的方法,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-09-09有關(guān)Java常見(jiàn)的誤解小結(jié)(來(lái)看一看)
下面小編就為大家?guī)?lái)一篇有關(guān)Java常見(jiàn)的誤解小結(jié)(來(lái)看一看)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-05-05springboot項(xiàng)目如何設(shè)置session的過(guò)期時(shí)間
這篇文章主要介紹了springboot項(xiàng)目如何設(shè)置session的過(guò)期時(shí)間,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-01-01springboot項(xiàng)目main函數(shù)啟動(dòng)的操作
這篇文章主要介紹了springboot項(xiàng)目main函數(shù)啟動(dòng)的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-06-06