自定义注解 + Redis 实现业务的幂等性

1.实现幂等性思路

实现幂等性有两种方式:

⭐ 1. 在数据库层面进行幂等性处理(数据库添加唯一约束).

例如:新增用户幂等性处理,username 字段可以添加唯一约束.

⭐ 2. 在应用程序层面进行幂等性处理.

而在应用程序方面进行幂等性处理,又有两种方式:

  • 通过 Spring AOP 方式实现幂等性判断(需要额外添加依赖).
  • 通过 Spring Boot 提供的拦截器实现幂等性判断.

例如:发表评论,同一个用户可以发表相同的评论,添加唯一约束不合适,放在程序层面处理.

2. 自定义注解 + Redis 实现业务幂等性

【实现思路】

  1. 创建自定义幂等性注解.

  2. 实现自定义幂等性注解的拦截器

    1. 创建拦截器,添加幂等性判断逻辑

    2. 定义幂等性判断的 ID(两种方式)

      1. 请求方携带唯一业务 ID

      2. 后端程序自行组织唯一业务 ID:当前用户 ID + 请求的数据(此处使用第二种)

  3. 配置拦截规则

  4. 使用自定义幂等性注解来保证业务的幂等性

2.1 自定义幂等性注解

/**
 * 自定义幂等性判断注解
 *
 * @author helong
 */
@Target(ElementType.METHOD) // 方法注解
@Retention(RetentionPolicy.RUNTIME)  // 程序运行期间有效
public @interface Idempotent {

    /**
     * 幂等性判断的时效
     *
     * @return
     */
    int time() default 60;
}

2.2 实现自定义幂等性注解的拦截器

@Component
public class IdempotentInterceptor implements HandlerInterceptor {

    @Resource
    private ObjectMapper objectMapper;
    @Resource
    private StringRedisTemplate stringRedisTemplate;


    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        // 只处理控制器方法,而不处理其他类型的请求(如静态资源)
        if (handler instanceof HandlerMethod) {
            Method method = ((HandlerMethod) handler).getMethod();
            // 尝试获取方法上的 Idempotent 注解
            Idempotent idempotent = method.getAnnotation(Idempotent.class);
            if (ObjectUtil.isNotNull(idempotent)) {
                // 生成唯一业务 ID
                String id = createId(request);
                ValueOperations<String, String> ops = stringRedisTemplate.opsForValue();

                // 如果 Redis 中已存在相同的业务 ID,阻止重复提交
                if (ObjectUtil.isNotNull(ops.get(id))) {
                    response.setContentType("application/json;charset=UTF-8");
                    response.setCharacterEncoding("UTF-8");
                    String json = "{\"code\": 500, \"msg\": \"数据正在处理,请勿重复提交!\", \"data\": null}";
                    response.getWriter().write(json);
                    return false;
                } else {
                    // 如果 Redis 中不存在相同的业务 ID,存储这个 ID 并设置过期时间
                    ops.set(id, Boolean.TRUE.toString(), idempotent.time(), TimeUnit.SECONDS);
                    return true;
                }
            }
        }
        // 如果不是 HandlerMethod 实例或没有 Idempotent 注解,继续处理请求
        return HandlerInterceptor.super.preHandle(request, response, handler);
    }

    /**
     * 生成幂等性 Id -> md5(用户ID + 请求参数)
     *
     * @param request
     */
    private String createId(HttpServletRequest request) throws JsonProcessingException {
    // 获取当前用户 ID
    SecurityUserDetails userDetails = SecurityUtil.getCurrentUser();
    Long uid = ObjectUtil.isNotNull(userDetails) ? userDetails.getUid() : NumberUtils.LONG_ZERO;
    // 将请求参数转换为 JSON 字符串
    String requestParam = objectMapper.writeValueAsString(request.getParameterMap());
    return SecureUtil.md5(uid + requestParam);
}
}

2.3 配置拦截规则

@Configuration
public class WebConfig implements WebMvcConfigurer {

    /**
     * 注入自定义拦截器
     */
    @Resource
    private IdempotentInterceptor idempotentInterceptor;

    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(idempotentInterceptor)
                // 拦截所有的请求
                .addPathPatterns("/**")
                // 放行静态资源
                .excludePathPatterns("/index.html")
                .excludePathPatterns("/login.html")
                .excludePathPatterns("/image/**")
                .excludePathPatterns("/js/**")
                .excludePathPatterns("/layui/**");
    }
}

此处也可以不需要放行静态资源,因为上一步的自定义幂等性注解拦截器的逻辑里,第一个 if 就相当于放行了静态资源。

2.4 使用自定义幂等性注解

@PostMapping("/add")
@Idempotent
public ResponseEntity addComment(@Validated Comment comment) {
    comment.setUid(SecurityUtil.getCurrentUser().getUid());
    boolean result = commentService.save(comment);
    return result ? ResponseEntity.success(Boolean.TRUE) : ResponseEntity.fail("评论失败");
}

就拿发表评论来看,添加完自定义幂等性注解后,来到前端页面尝试在 1 分钟内,使用相同的用户,发表相同的评论:

PS:Security 用户对象,获取当前登录用户的代码,请参照这篇文章:SpringSecurity + JWT 实现登录认证-CSDN博客

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Master_hl

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值