Java はフロントエンドでの繰り返しリクエストをどのように制限しますか?

WBOY
リリース: 2023-05-14 22:58:04
転載
1407 人が閲覧しました

背景と目的

フロントエンド ページが停止し、ユーザーが操作ボタンを繰り返しクリックすると、バックエンド インターフェイスが短期間に複数回送信されます

実装手順

アスペクトを設定し、指定時間内にインターフェイスが繰り返し呼び出されないようにアノテーションを追加します

インターフェイスを設定します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:";
}
ログイン後にコピー
ログイン後にコピー

Use

    @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. コントローラーインターフェイスの定義

	@GetMapping("/query")
    @LocalLock(key = "param:arg[0]")
    public String query (@RequestParam("abc") String abc) {
        return "ok";
    }
ログイン後にコピー
First 最初の呼び出しの結果:

Java はフロントエンドでの繰り返しリクエストをどのように制限しますか?

2 番目の呼び出しの結果:

Java はフロントエンドでの繰り返しリクエストをどのように制限しますか?

以上がJava はフロントエンドでの繰り返しリクエストをどのように制限しますか?の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

関連ラベル:
ソース:yisu.com
このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
人気のチュートリアル
詳細>
最新のダウンロード
詳細>
ウェブエフェクト
公式サイト
サイト素材
フロントエンドテンプレート