springboot防止表单重复提交

打印 上一主题 下一主题

主题 984|帖子 984|积分 2952

第一种方法:单个防止
在Spring Boot应用中使用Redis来防止表单的重复提交,可以通过以下几个步骤来实现:
步骤 1: 添加依赖

确保你的项目中添加了Spring Boot Starter Data Redis和Spring Boot Starter Web依赖。在pom.xml文件中添加以下依赖:
  1. <dependencies>
  2.     <dependency>
  3.         <groupId>org.springframework.boot</groupId>
  4.         <artifactId>spring-boot-starter-data-redis</artifactId>
  5.     </dependency>
  6.     <dependency>
  7.         <groupId>org.springframework.boot</groupId>
  8.         <artifactId>spring-boot-starter-web</artifactId>
  9.     </dependency>
  10. </dependencies>
复制代码
步骤 2: 配置Redis

在application.properties或application.yml中配置Redis服务器的信息:
  1. spring.redis.host=localhost
  2. spring.redis.port=6379
复制代码
步骤 3: 创建一个工具类处理Redis操作

创建一个工具类来封装Redis的一些常用操作,比如设置键值对、获取键值对等。
  1. import org.springframework.beans.factory.annotation.Autowired;
  2. import org.springframework.data.redis.core.StringRedisTemplate;
  3. import org.springframework.stereotype.Component;
  4. @Component
  5. public class RedisUtil {
  6.     @Autowired
  7.     private StringRedisTemplate stringRedisTemplate;
  8.     public void set(String key, String value) {
  9.         stringRedisTemplate.opsForValue().set(key, value);
  10.     }
  11.     public String get(String key) {
  12.         return stringRedisTemplate.opsForValue().get(key);
  13.     }
  14.     public void delete(String key) {
  15.         stringRedisTemplate.delete(key);
  16.     }
  17. }
复制代码
步骤 4: 实现防重逻辑

在Controller中使用上面的工具类来实现防重逻辑。通常的做法是在哀求开始时查抄Redis中是否存在该哀求的唯一标识(如用户ID + 时间戳),如果不存在则允许哀求继承,并将此标识存入Redis;如果存在,则直接返回错误信息。
  1. import org.springframework.beans.factory.annotation.Autowired;
  2. import org.springframework.web.bind.annotation.PostMapping;
  3. import org.springframework.web.bind.annotation.RestController;
  4. @RestController
  5. public class MyController {
  6.     @Autowired
  7.     private RedisUtil redisUtil;
  8.     @PostMapping("/submit")
  9.     public String submit(@RequestParam("userId") String userId) {
  10.         // 使用用户ID和时间戳作为key,防止并发下的重复提交
  11.         String key = "form_submit_" + userId + "_" + System.currentTimeMillis();
  12.         if (redisUtil.get(key) != null) {
  13.             return "请求重复,请勿重复提交!";
  14.         } else {
  15.             // 执行业务逻辑
  16.             // ...
  17.             // 将key存入Redis并设置过期时间
  18.             redisUtil.set(key, "1", 5, TimeUnit.MINUTES);
  19.             return "提交成功!";
  20.         }
  21.     }
  22. }
复制代码
请注意,在现实应用中,你可能需要根据详细需求调整上述代码,比如更改存储在Redis中的键的生成方式、设置更合理的逾期时间等。别的,为了提高性能和避免并发问题,可以考虑使用Redis的原子操作,例如SETNX下令。
第二种方法:aop全局方法
使用AOP(面向切面编程)来统一处理防重复提交是一种很好的实践,它能减少重复代码并保持业务逻辑的清晰性。下面是如何使用Spring AOP来实现这一功能的示例。
首先,你需要在项目中添加Spring AOP的支持,通常情况下Spring Boot项目已经默认包含AOP支持。
步骤 1: 创建切面类

创建一个切面类,用于界说防重提交的逻辑。
  1. import org.aspectj.lang.ProceedingJoinPoint;
  2. import org.aspectj.lang.annotation.Around;
  3. import org.aspectj.lang.annotation.Aspect;
  4. import org.springframework.beans.factory.annotation.Autowired;
  5. import org.springframework.data.redis.core.StringRedisTemplate;
  6. import org.springframework.stereotype.Component;
  7. import java.util.concurrent.TimeUnit;
  8. @Aspect
  9. @Component
  10. public class AntiDuplicateSubmissionAspect {
  11.     private static final String KEY_PREFIX = "anti_duplicate_";
  12.     @Autowired
  13.     private StringRedisTemplate stringRedisTemplate;
  14.     @Around("@annotation(antiDuplicateSubmission)")
  15.     public Object antiDuplicateSubmit(ProceedingJoinPoint joinPoint, AntiDuplicateSubmission antiDuplicateSubmission) throws Throwable {
  16.         // 从注解中获取参数名
  17.         String paramName = antiDuplicateSubmission.value();
  18.         // 获取方法参数
  19.         Object[] args = joinPoint.getArgs();
  20.         // 假设参数位置已知,或者通过参数名获取参数值
  21.         String userId = (String) args[0];
  22.         // 生成唯一标识符
  23.         String key = KEY_PREFIX + userId + "_" + System.currentTimeMillis();
  24.         // 检查是否已经存在
  25.         if (stringRedisTemplate.hasKey(key)) {
  26.             throw new RuntimeException("请求重复,请勿重复提交!");
  27.         }
  28.         // 设置过期时间,例如5分钟
  29.         stringRedisTemplate.opsForValue().set(key, "1", 5, TimeUnit.MINUTES);
  30.         try {
  31.             return joinPoint.proceed();
  32.         } finally {
  33.             // 清理key,这一步在大多数情况下可选,因为设置了过期时间
  34.             stringRedisTemplate.delete(key);
  35.         }
  36.     }
  37. }
复制代码
步骤 2: 创建自界说注解

创建一个自界说注解,用于标记需要防重提交的控制器方法。
  1. import java.lang.annotation.ElementType;
  2. import java.lang.annotation.Retention;
  3. import java.lang.annotation.RetentionPolicy;
  4. import java.lang.annotation.Target;
  5. @Target(ElementType.METHOD)
  6. @Retention(RetentionPolicy.RUNTIME)
  7. public @interface AntiDuplicateSubmission {
  8.     String value();
  9. }
复制代码
步骤 3: 应用注解

在需要防重提交的Controller方法上应用这个注解。
  1. import org.springframework.web.bind.annotation.PostMapping;
  2. import org.springframework.web.bind.annotation.RestController;
  3. @RestController
  4. public class MyController {
  5.     @PostMapping("/submit")
  6.     @AntiDuplicateSubmission(value = "userId")
  7.     public String submit(@RequestParam("userId") String userId) {
  8.         // 执行业务逻辑
  9.         // ...
  10.         return "提交成功!";
  11.     }
  12. }
复制代码
这样,每次有哀求到达带有@AntiDuplicateSubmission注解的方法时,AOP切面会先执行防重查抄,如果发现重复提交,则制止业务逻辑执行并抛出非常。
注意,这里的AntiDuplicateSubmission注解的value属性应该与哀求参数名匹配,以便正确地获取到用户ID或其他用于生成唯一标识符的参数。同时,使用System.currentTimeMillis()生成的时间戳作为一部门唯一标识符可能在高并发场景下存在并发问题,你可能需要一个更安全的唯一生成策略,例如UUID大概结适用户ID、哀求时间以及随机数等生成一个更复杂的唯一标识符。

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

大号在练葵花宝典

金牌会员
这个人很懒什么都没写!
快速回复 返回顶部 返回列表