Java 限制前端重復請求的實例代碼

背景及用途

前端頁面出現卡頓,用戶反復點擊操作按鈕,導致後臺接口短時間內多次提交

實現步驟

設置切面,增加註解,導致在規定時間內該接口不可重復調用

設置一個接口 NoRepeatSubmit

import java.lang.annotation.*;
/**
 * xzj_2022_8_2
 * 重復請求限制切面
 */
@Target(ElementType.METHOD) //註解放置的目標位置,METHOD是可註解在方法級別上
@Retention(RetentionPolicy.RUNTIME) //註解在哪個階段執行
@Documented //生成文檔
public @interface NoRepeatSubmit {
    String name() default "name:";
}

實現類

import java.lang.annotation.*;
/**
 * xzj_2022_8_2
 * 重復請求限制切面
 */
@Target(ElementType.METHOD) //註解放置的目標位置,METHOD是可註解在方法級別上
@Retention(RetentionPolicy.RUNTIME) //註解在哪個階段執行
@Documented //生成文檔
public @interface NoRepeatSubmit {
    String name() default "name:";
}

使用

    @GetMapping(value = "/test")
    @NoRepeatSubmit
    public void test() {
        System.out.println("test");
        }

補充:下面看下java防止前端重復提交

JAVA利用自定義本地鎖解決重復提交的問題

1.引入jar包

<dependency>
            <groupId>com.google.guava</groupId>
            <artifactId>guava</artifactId>
            <version>31.1-jre</version>
        </dependency>

2.自定義本地鎖

package com.hzt.listener;
import java.lang.annotation.*;
/**
 * 自定義-控制重復提交鎖
 */
@Target(ElementType.METHOD)     //作用於方法
@Retention(RetentionPolicy.RUNTIME)     //運行時有效
@Documented
@Inherited
public @interface LocalLock {

    String key() default "";
}

3.自定義註解切面 (aop攔截器實現)

import com.google.common.cache.Cache;
import com.google.common.cache.CacheBuilder;
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.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.context.annotation.Configuration;

import java.util.concurrent.TimeUnit;

/**
 * @Desc: 自定義註解攔截器
 * @Author: zmk
 * @Date: 2022/4/2
 */
@Aspect
@Configuration
public class LockMethodInterceptor {
    private final Logger log = LoggerFactory.getLogger(LockMethodInterceptor.class);


    private static final Cache<String, Object> CACHES = CacheBuilder.newBuilder()
            //最大緩存數
            .maximumSize(1000)
            //設置過期時間
            .expireAfterWrite(3, TimeUnit.SECONDS)
            .build();



    @Around(value = "@annotation(localLock)")
    public Object interceptor (ProceedingJoinPoint point, LocalLock localLock) {

        //localLock.key() 這個是獲取controller的key屬性, point.getArgs()獲取key的值
        String key = getKey(localLock.key(), point.getArgs());
        if (StringUtils.isNotBlank(key)) {
            if (CACHES.getIfPresent(key) != null) {
                throw new RuntimeException("請勿重復提交");
            }
            //如果是第一次請求, 將key放入緩存
            CACHES.put(key, key);
        }

        try {
            return point.proceed();
        } catch (Throwable throwable) {
            throw new RuntimeException("服務器異常");
        } finally {
            //標記為無效
//            CACHES.invalidate(key);
        }

    }
    
    /**
     *
     * key 生成策略
     * @param key   key表達式
     * @param args  參數
     * @return  生成的key
     */
    private String getKey(String key, Object[] args) {
        for (int i = 0; i < args.length; i++) {
            key = key.replace("arg[" + i + "]", args[i].toString());
        }
        return key;
    }

4.定義controller接口

	@GetMapping("/query")
    @LocalLock(key = "param:arg[0]")
    public String query (@RequestParam("abc") String abc) {
        return "ok";
    }

第一次調用結果:

第二次調用結果:

到此這篇關於Java 限制前端重復請求的文章就介紹到這瞭,更多相關Java 限制重復請求內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: