当前位置: 首页 > news >正文

SpringBoot 防刷 重复提交问题 重复点击问题 注解 RequestParam RequestBody

下订单要调用第三方接口,操作比较久,用户很容易点击两次,前端后端要做防刷处理

注解

package net.digital.smart.config;import java.lang.annotation.*;/*** 用于防刷限流的注解* 默认是5秒内只能调用一次*/
@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface RateLimit {/*** 限流的key*/String key() default "RateLimit:";/*** 周期,单位是秒*/int cycle() default 5;/*** 默认提示信息*/String msg() default "正在处理中,请您稍微歇一会。";/*** 请求参数名称,用于获取请求中的参数值 RequestParam为参数index,RequestBody为参数名*/String paramName() default "";/*** 请求参数类型*/String paramType() default "";
}

实现类

package net.digital.smart.config;import cn.hutool.core.util.ArrayUtil;
import cn.hutool.core.util.ObjectUtil;
import cn.hutool.core.util.StrUtil;
import cn.hutool.json.JSONUtil;
import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.JSONArray;
import com.alibaba.fastjson.JSONObject;
import lombok.extern.slf4j.Slf4j;
import net.digital.common.core.exception.BizErrorException;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.reflect.MethodSignature;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.core.DefaultParameterNameDiscoverer;
import org.springframework.core.ParameterNameDiscoverer;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Component;
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.multipart.MultipartFile;import javax.servlet.http.HttpServletRequest;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.lang.reflect.Method;
import java.lang.reflect.Parameter;
import java.nio.charset.StandardCharsets;
import java.util.Arrays;
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.TimeUnit;@Aspect
@Component
@Slf4j
public class RateLimitInterceptor {@Autowiredprivate RedisTemplate<String, Object> redisTemplate;@Before("@annotation(net.digital.smart.config.RateLimit)")public void before(JoinPoint point) {// 获取方法上的RateLimit注解MethodSignature signature = (MethodSignature) point.getSignature();Method method = signature.getMethod();RateLimit rateLimit = method.getAnnotation(RateLimit.class);if (rateLimit == null) {return;}try {// 生成限流key,格式为:类名+方法名+IP+自定义key+参数值String key = generateKey(rateLimit, method, point.getArgs());log.info("接口防刷key:" + key);// 检查是否超过限流次数if (!tryAcquire(key, rateLimit.cycle())) {throw new BizErrorException(rateLimit.msg());}} catch (Exception e) {throw new BizErrorException(e.getMessage());}}private String generateKey(RateLimit rateLimit, Method method, Object[] args) throws IOException {StringBuilder keyBuilder = new StringBuilder();String paramValue = getParameterValue(args, rateLimit.paramName(),rateLimit.paramType());keyBuilder.append(rateLimit.key()).append(paramValue).append(":").append(method.getDeclaringClass().getName()).append(":").append(method.getName());return keyBuilder.toString();}private boolean tryAcquire(String key, int cycle) {// 使用Redis的setIfAbsent实现原子操作Boolean success = redisTemplate.opsForValue().setIfAbsent(key, 1, cycle, TimeUnit.SECONDS);return success != null && success;}private String getParameterValue(Object[] args, String paramName,String paramType) {if (ArrayUtil.isEmpty(args)) {throw new BizErrorException("获取防刷key失败");}String argsStr = JSONUtil.toJsonStr(args);log.info("获取防刷key-argsStr=>{}",argsStr);if (StrUtil.isBlank(argsStr)) {throw new BizErrorException("获取防刷key失败");}JSONArray jsonArray = JSONObject.parseArray(argsStr);if (jsonArray.isEmpty()){throw new BizErrorException("获取防刷key失败");}if (StrUtil.equals(paramType,"RequestBody")){JSONObject jsonObject = jsonArray.getJSONObject(0);if (ObjectUtil.isEmpty(jsonObject)) {throw new BizErrorException("获取防刷key失败");}log.info("防刷jsonObject=>{}", JSON.toJSONString(jsonObject));String uuid = jsonObject.getString(paramName);if (StrUtil.isNotBlank(uuid)) {return uuid;}throw new BizErrorException("获取防刷key失败");} else if ((StrUtil.equals(paramType, "RequestParam"))) {for (int i = 0; i < jsonArray.size(); i++) {if (StrUtil.equals(String.valueOf(i), paramName)) {return jsonArray.getString(i);}}}throw new BizErrorException("获取防刷key失败");}
}

使用方法

    /*** 锁号|预约挂号*/@RateLimit(paramName = "contactId",paramType = "RequestBody")@PostMapping("/order/create")@Operation(description = "锁号|预约挂号", summary = "锁号|预约挂号")@PreAuthorize("@pms.hasPermission('pms_member_biz')")public R createOrder(@Validated @RequestBody RegOrderDto regOrderDto) {if (ObjectUtil.isEmpty(regOrderDto.getDeptId())) {throw new BizErrorException("deptId不能为空");}if (StrUtil.isEmpty(regOrderDto.getAvailableNumStr())) {throw new BizErrorException("availableNumStr不能为空");}RegisteredHistory order = registeredService.createOrder(regOrderDto);if (null == order) {throw new BizErrorException(ErrorEnum.REG_APPLY_FAILED);}// 同步挂号信息到第三方return R.ok(order);}/*** 取消锁号|取消预约订单*/@RateLimit(paramName = "1",paramType = "RequestParam")@PostMapping("/order/cancel")@Operation(description = "取消锁号|取消预约订单", summary = "取消锁号|取消预约订单")@PreAuthorize("@pms.hasPermission('pms_member_biz')")public R cancelOrder(@RequestParam(value = "regId") Long regId,@RequestParam(value = "contactId", required = false) Long contactId) {if (null == contactId || null == regId) {throw new BizErrorException(ErrorEnum.SYSTEM_VARIABLE_INVALID);}Long mechanismId = RequestCtx.getHeader(request).getMechanismId();CommonCardInfo commonCardInfo = getCommonCardInfo(SecurityUtils.getUser().getId(), contactId, mechanismId);boolean cancel = smtRegisteredRecordService.cancel(regId, CancelTypeEnum.USER_CANCEL.getType(), commonCardInfo.getHisPatientId());if (cancel) {return R.ok();}return R.failed();}
http://www.lryc.cn/news/575887.html

相关文章:

  • 深度学习框架入门指南:PyTorch 核心实战
  • 临床项目计划框架
  • debian挂载新硬盘后不识别怎么办?
  • 【 MyBatis-Plus | 精讲 】
  • Spring Boot 项目实训 - 图书信息网站
  • 分布式ID生成SnowflakeId雪花算法和百度UidGenerator工具类
  • 微信小程序跳转传参方式
  • 链表最终章——双向链表及其应用
  • Stable Diffusion入门-ControlNet 深入理解-第三课:结构类模型大揭秘——深度、分割与法线贴图
  • 【向上教育】结构化面试开口秘籍.pdf
  • 【江科大】STM32F103C8T6 + TB6612 + N20编码器减速电机《03-增量式PID定速控制》(增量式PID,定时器输入捕获,定时器编码器)
  • 动手学Python:从零开始构建一个“文字冒险游戏”
  • Fiddler中文版抓包工具在跨域与OAuth调试中的深度应用
  • 电子电气架构 --- 车联网技术简介
  • 什么是国际期货?期货交易平台搭建
  • 在反向代理环境下精准获取客户端真实 IP 的最佳实践
  • Java项目:基于SSM框架实现的宠物综合服务平台管理系统【ssm+B/S架构+源码+数据库+毕业论文+开题报告】
  • 论分布式设计
  • 学习设计模式《十五》——模板方法模式
  • Python打卡:Day39
  • LLM驱动开发:正在重塑软件工程的下一场革命
  • Moxa 加入 The Open Group 的开放流程自动化™论坛,推动以开放、中立标准强化工业自动化
  • uniapp处理后端返回的html字符串
  • Redis-zset有序集合
  • 什么是DNS缓存投毒?有哪些防御措施?
  • mac 安装python,切换python版本
  • 聚铭网络入选嘶吼《中国网络安全细分领域产品名录》“云平台安全管理”与“态势感知”双领域TOP10
  • 【C++】责任链模式
  • VSCode中创建和生成动态库项目
  • CSS3实现同心圆效果