SpringBoot开发——Spring Boot中实现订单30分钟自动取消的策略

文章目录

  • 简介
  • 方案一:定时任务
  • 方案二:延迟队列
  • 方案三:Redis过期事件
    • Redis的过期事件配置
  • 总结

简介

在电商和其他涉及到在线支付的应用中,通常需要实现一个功能:如果用户在生成订单后的一定时间内未完成支付,系统将自动取消该订单。本文将详细介绍基于 Spring Boot 框架实现订单30分钟内未支付自动取消的几种方案,并提供实例代码。

方案一:定时任务

利用 Spring Boot 中的 @Scheduled 注解,我们可以轻松地实现定时任务。该任务将周期性地扫描数据库,检查未支付的订单,如果订单生成30分钟未支付,则自动取消。

java 复制代码
@Component
public class OrderCancelSchedule {
    @Autowired
    private OrderService orderService;
    
    @Scheduled(cron = "0 0/1 * * * ?")
    public void cancelUnpaidOrders() {
        List<Order> unpaidOrders = orderService.getUnpaidOrders();
        unpaidOrders.forEach(order -> {
            if (order.getCreationTime().plusMinutes(30).isBefore(LocalDateTime.now())) {
                orderService.cancelOrder(order.getId());
            }
        });
    }
}

方案二:延迟队列

使用消息队列(如 RabbitMQ)的延迟队列功能,当订单生成时将订单ID推送到延迟队列,设置30分钟后过期,过期后消费该消息,取消订单。

java 复制代码
@Service
public class OrderService {
    @Autowired
    private RabbitTemplate rabbitTemplate;
    
    public void createOrder(Order order) {
        saveOrderToDB(order);
        rabbitTemplate.convertAndSend("orderDelayExchange", "orderDelayKey", order.getId(), message -> {
            message.getMessageProperties().setDelay(30 * 60 * 1000);  // 设置30分钟延迟
            return message;
        });
    }
}

@Component
@RabbitListener(queues = "orderDelayQueue")
public class OrderDelayConsumer {
    @Autowired
    private OrderService orderService;

    @RabbitHandler
    public void process(String orderId) {
        orderService.cancelOrder(orderId);
    }
}

方案三:Redis过期事件

利用 Redis 的键过期事件功能,当订单生成时在 Redis 中存储一个键,设置30分钟过期。键过期时,通过 Redis 的过期事件通知功能触发订单取消操作。

java 复制代码
@Service
public class OrderService {
    @Autowired
    private StringRedisTemplate redisTemplate;
    
    public void createOrder(Order order) {
        saveOrderToDB(order);
        redisTemplate.opsForValue().set("order:" + order.getId(), order.getId(), 30, TimeUnit.MINUTES);
    }

    public void onOrderKeyExpired(String orderId) {
        cancelOrder(orderId);
    }
}

Redis的过期事件配置

Redis的键过期通知是一种典型的发布-订阅模式。你需要确保Redis服务器开启了键空间通知功能。

具体步骤

首先,确保 Redis 的配置文件(通常是 redis.conf)中开启了键空间通知功能。你可以通过添加或修改如下配置实现:

bash 复制代码
notify-keyspace-events "Ex"

这里的 "Ex" 表示只监听键过期事件。如果需要监听其他类型的事件,可以参考 Redis 官方文档进行配置。

然后,在 Spring Boot 应用中,使用 RedisMessageListenerContainer 来订阅 Redis 的键过期事件,并指定回调方法进行处理。示例如下:

java 复制代码
@Configuration
public class RedisConfig {
    @Autowired
    private RedisConnectionFactory redisConnectionFactory;
    
    @Bean
    RedisMessageListenerContainer container() {
        RedisMessageListenerContainer container = new RedisMessageListenerContainer();
        container.setConnectionFactory(redisConnectionFactory);
        container.addMessageListener(new MessageListener() {
            @Override
            public void onMessage(Message message, byte[] pattern) {
                String expiredKey = message.toString();
                if (expiredKey.startsWith("order:")) {
                    String orderId = expiredKey.split(":")[1];
                    // 处理订单取消逻辑
                }
            }
        }, new PatternTopic("__keyevent@*__:expired"));
        return container;
    }
}

在这个示例中,"__keyevent@*__:expired" 是一个模式匹配的主题,它可以匹配所有数据库的键过期事件。当一个键过期时,onMessage 方法会被调用,你可以在这里加入你的逻辑来处理订单的超时取消。

总结

以上三种方案都可以实现订单在30分钟内未支付则自动取消的需求。根据实际业务需求、系统负载和其他因素,可以选择最适合自己系统的实现方案。每种方案都有其优缺点,需要根据具体情况权衡。

相关推荐
汤姆yu2 小时前
基于springboot的尿毒症健康管理系统
java·spring boot·后端
暮色妖娆丶2 小时前
Spring 源码分析 单例 Bean 的创建过程
spring boot·后端·spring
biyezuopinvip3 小时前
基于Spring Boot的企业网盘的设计与实现(任务书)
java·spring boot·后端·vue·ssm·任务书·企业网盘的设计与实现
JavaGuide4 小时前
一款悄然崛起的国产规则引擎,让业务编排效率提升 10 倍!
java·spring boot
figo10tf4 小时前
Spring Boot项目集成Redisson 原始依赖与 Spring Boot Starter 的流程
java·spring boot·后端
zhangyi_viva4 小时前
Spring Boot(七):Swagger 接口文档
java·spring boot·后端
橙露4 小时前
Spring Boot 核心原理:自动配置机制与自定义 Starter 开发
java·数据库·spring boot
程序员敲代码吗5 小时前
Spring Boot与Tomcat整合的内部机制与优化
spring boot·后端·tomcat
NuageL5 小时前
原始Json字符串转化为Java对象列表/把中文键名变成英文键名
java·spring boot·json
jzheng86105 小时前
Spring Boot(快速上手)
java·spring boot·后端