java接口防重提交的處理方法
1.什么是接口防重?
在一定的時間內多次請求同一接口,同一參數(shù)。由于請求是健康請求,會執(zhí)行正常的業(yè)務邏輯,從而產(chǎn)生大量的廢數(shù)據(jù)。
2.問題的產(chǎn)生及引發(fā)的問題
舉一個最簡單的例子:日常開發(fā)中crud在業(yè)務系統(tǒng)中普遍存在,在服務端沒有做任何處理,客戶端沒有做節(jié)流、防抖等限流操作時,同一秒一個用戶點了兩次新增按鈕,導致數(shù)據(jù)庫中存在同樣兩條數(shù)據(jù),其結果可想而知,同理修改、刪除同樣的道理;查詢本身具有冪等性,但是在同一秒鐘同樣的操作,查詢多次和一次,有區(qū)別嗎?區(qū)別大了去了,不談用戶體驗如何,光是網(wǎng)絡開銷、流量占用、帶給服務器的壓力等等,生產(chǎn)中一點小的問題,如何不及時處理,可能會引發(fā)災難性bug。
3.處理方法
- 第一種:前臺在請求接口的時候,傳遞一個唯一值,然后在對應接口判斷該唯一值,在一定的時間內是否被消費過
- 第二種:采用Spring AOP理念,實現(xiàn)請求的切割,在請求執(zhí)行到某個方法或某層時候,開始攔截進行,獲取該請求的參數(shù),用戶信息,請求地址,存入redis中并放置過期時間,進行防重(推薦使用)
4.談談以下兩種處理方法的利弊
- 第一種:局限性太高,前臺必須傳遞一個唯一值,就算請求到達指定后臺服務,寫一個攔截器,需要配置太多不需要攔截的方法,也許你會說,可以攔截有規(guī)則的請求地址,這樣真的好嗎?
- 第二種(推薦):采用AOP面向切面編程的思想,在不污染源代碼的情況下,進行增強功能,切入到要防重的接口上,實現(xiàn)統(tǒng)一防重處理、業(yè)務解耦。此處采用AOP + 自定義注解,靈活實現(xiàn)防重功能。
5.具體代碼(采用第二種)
注解類
import java.lang.annotation.*;
/**
?* 防重
?* @date 2020/8/12
?* @return
?*/
//標識該注解用于方法上
@Target({ElementType.METHOD})
//申明該注解為運行時注解,編譯后改注解不會被遺棄
@Retention(RetentionPolicy.RUNTIME)
//javadoc工具記錄
@Documented
public @interface PreventSubmit?
{
}切面類
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;
/**
?* 防重復提交
?* @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:";
? ? /**
? ? ?* 失效時間
? ? ?*/
? ? 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 {
? ? ? ? ? ? //獲取目標方法的參數(shù)
? ? ? ? ? ? Object[] args = joinPoint.getArgs();
? ? ? ? ? ? //獲取當前request請求
? ? ? ? ? ? RequestAttributes requestAttributes = RequestContextHolder.getRequestAttributes();
? ? ? ? ? ? HttpServletRequest request = (HttpServletRequest) requestAttributes.resolveReference(RequestAttributes.REFERENCE_REQUEST);
? ? ? ? ? ? //獲取請求地址
? ? ? ? ? ? String requestUri = request.getRequestURI();
? ? ? ? ? ? //獲取用戶ID
? ? ? ? ? ? Long userId = null;
? ? ? ? ? ? try {
? ? ? ? ? ? ? ? userId = TokenUtil.getUserId(request);
? ? ? ? ? ? }catch (Exception e){}
? ? ? ? ? ? //拼接鎖前綴,采用同一方法,同一用戶,同一接口
? ? ? ? ? ? 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);
? ? ? ? ? ? /**
? ? ? ? ? ? ?* 對同一方法同一用戶同一參數(shù)加鎖,即使獲取不到用戶ID,每個用戶請求數(shù)據(jù)也會不一致,不會造成接口堵塞
? ? ? ? ? ? ?*/
? ? ? ? ? ? lock = this.redissonClient.getLock(lockPrefix);
? ? ? ? ? ? lock.lock();
? ? ? ? ? ? String flag = this.stringRedisTemplate.opsForValue().get(redisPrefix);
? ? ? ? ? ? if(StringUtils.isNotEmpty(flag)){
? ? ? ? ? ? ? ? throw new AppException("您當前的操作太頻繁了,請稍后再試!");
? ? ? ? ? ? }
? ? ? ? ? ? //存入redis,設置失效時間
? ? ? ? ? ? this.stringRedisTemplate.opsForValue()
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?.set(redisPrefix,redisPrefix,INVALID_NUMBER, TimeUnit.SECONDS);
? ? ? ? ? ? //執(zhí)行目標方法
? ? ? ? ? ? Object result = joinPoint.proceed(args);
? ? ? ? ? ? return result;
? ? ? ? }finally {
? ? ? ? ? ? if(lock != null){
? ? ? ? ? ? ? ? lock.unlock();
? ? ? ? ? ? }
? ? ? ? }
? ? }
}到此這篇關于java接口防重提交的處理方法的文章就介紹到這了,更多相關java接口防重提交內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
MybatisPlus自定義Sql實現(xiàn)多表查詢的示例
這篇文章主要介紹了MybatisPlus自定義Sql實現(xiàn)多表查詢的示例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-08-08
在Springboot中Mybatis與Mybatis-plus的區(qū)別詳解
MyBatis是一個優(yōu)秀的持久層框架,它對JDBC的操作數(shù)據(jù)庫的過程進行封裝,MyBatisPlus (簡稱 MP)是一個 MyBatis的增強工具,在 MyBatis 的基礎上只做增強不做改變,為簡化開發(fā)、提高效率而生,本文將給大家介紹了在Springboot中Mybatis與Mybatis-plus的區(qū)別2023-12-12

