首頁 > 軟體

Java 限制前端重複請求的範例程式碼

2022-08-03 22:04:12

背景及用途

前端頁面出現卡頓,使用者反覆點選操作按鈕,導致後臺介面短時間內多次提交

實現步驟

設定切面,增加註解,導致在規定時間內該介面不可重複呼叫

設定一個介面 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 限制重複請求內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com