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

Spring Boot中利用Redis解决接口幂等性问题

在Spring Boot中利用Redis解决接口幂等性问题,可以通过以下步骤实现:


1. 核心思路

  • 唯一标识:每次请求生成唯一ID(如UUID或业务标识),作为Redis的Key。
  • 原子操作:使用Redis的SETNX(SET if Not Exists)命令,确保同一请求只能执行一次。
  • 过期机制:为Key设置合理过期时间,避免无效数据堆积。

2. 实现步骤

2.1 添加依赖
<!-- Spring Boot Starter Data Redis -->
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
2.2 配置RedisTemplate
@Configuration
public class RedisConfig {
    @Bean
    public RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory factory) {
        RedisTemplate<String, Object> template = new RedisTemplate<>();
        template.setConnectionFactory(factory);
        template.setKeySerializer(new StringRedisSerializer());
        template.setValueSerializer(new GenericJackson2JsonRedisSerializer());
        return template;
    }
}
2.3 定义幂等性注解
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Idempotent {
    String keyPrefix() default "idempotent:";
    long expireTime() default 5000; // 过期时间(毫秒)
}
2.4 AOP切面处理
@Aspect
@Component
public class IdempotentAspect {
    @Autowired
    private RedisTemplate<String, Object> redisTemplate;

    @Around("@annotation(idempotent)")
    public Object handleIdempotent(ProceedingJoinPoint joinPoint, Idempotent idempotent) throws Throwable {
        String uniqueKey = generateUniqueKey(joinPoint, idempotent.keyPrefix());
        long expireTime = idempotent.expireTime();

        // 尝试设置Redis Key(原子操作)
        Boolean isSet = redisTemplate.opsForValue().setIfAbsent(uniqueKey, "LOCK", expireTime, TimeUnit.MILLISECONDS);
        if (isSet == null || !isSet) {
            throw new RuntimeException("重复请求,请稍后再试");
        }

        try {
            return joinPoint.proceed();
        } finally {
            // 业务完成后可选延长过期时间或保留原设置
            // redisTemplate.expire(uniqueKey, 60, TimeUnit.SECONDS);
        }
    }

    private String generateUniqueKey(ProceedingJoinPoint joinPoint, String prefix) {
        // 从请求参数或Header中提取唯一ID(示例从参数获取)
        Object[] args = joinPoint.getArgs();
        String requestId = (String) Arrays.stream(args)
                .filter(arg -> arg instanceof String && ((String) arg).startsWith("req_"))
                .findFirst()
                .orElse(UUID.randomUUID().toString());
        return prefix + requestId;
    }
}
2.5 控制器使用示例
@RestController
public class OrderController {
    @PostMapping("/pay")
    @Idempotent(keyPrefix = "order:pay:", expireTime = 60000)
    public ResponseEntity<String> payOrder(@RequestParam String orderId, @RequestParam String reqId) {
        // 业务逻辑(如扣款、更新订单状态)
        return ResponseEntity.ok("支付成功");
    }
}

3. 关键点说明

  1. 唯一ID生成

    • 客户端生成唯一reqId(如UUID),或服务端根据业务参数生成(如userId+orderId)。
    • 避免使用时间戳,防止碰撞。
  2. 过期时间设置

    • 根据业务耗时设置合理过期时间,避免因业务未完成导致Key提前过期。
  3. 异常处理

    • 业务异常需回滚操作,但幂等性Key保留,防止重复提交。
    • 可结合@Transactional管理事务与Redis操作的一致性。
  4. 高并发优化

    • 使用Redis集群提升吞吐量。
    • 对极高频请求可考虑本地缓存(如Caffeine)+ Redis双校验。

4. 扩展场景

  • 返回缓存结果:首次请求处理完成后,将结果存入Redis,后续相同请求直接返回缓存结果。
  • 结合数据库:关键操作在数据库层面添加唯一约束(如订单号唯一索引)。

通过上述方案,可有效避免重复请求导致的数据不一致问题,适用于支付、下单等高风险接口。


http://www.kler.cn/a/582807.html

相关文章:

  • html css 笔记
  • 访问权限控制、访问PHP站点
  • 13 | 实现统一的错误返回
  • Spring Boot 中实现全局 Token 验证的两种方式
  • Rule-Engine 使用介绍
  • upload-labs-master通关攻略(5~8)
  • DeepIn Wps 字体缺失问题
  • 【JavaWeb】快速入门——HTMLCSS
  • 在 Windows 11 下运行 OminiParse V2,详细教程【含问题解决细节】
  • pdf合并工具
  • [多线程]基于单例懒汉模式的线程池的实现
  • Redis 2025/3/9
  • nextJs在DOM视图中渲染未转为状态值的localStorage导致报错
  • mac 被禁用docker ui后,如何使用lima虚拟机启动docker
  • 【实战ES】实战 Elasticsearch:快速上手与深度实践-8.1.2近似最近邻(ANN)算法选型
  • 【Synchronized】不同的使用场景和案例
  • 信号处理之插值、抽取与多项滤波
  • 【C++】C++11新特性
  • ELK traceId 通过A服务调用B服务举例
  • Hive SQL 精进系列:COALESCE 手册