苍穹外卖day10(1)Spring Task、 订单状态定时处理


前言

Spring Task用于在指定时间处理某个业务逻辑,在本项目中体现在订单状态定时处理,包括“下单后未支付(订单超时自动取消)”、“订单一直处于派送中状态(订单需要自动更新成已完成)”。
实现的业务逻辑通过当前订单状态和下单时间去查询订单,对于满足上面两个条件的订单,设置一个时间让任务自动执行。


一、Spring Task

1.1 概述

Spring框架提供的任务调度工具,可以按照约定的时间自动执行某个代码逻辑。
应用场景

  • 信用卡每月还款提醒
  • 银行贷款每月还款提醒
  • 火车票售票系统处理未支付订单
  • 入职纪念日为用户发送通知
    只要是需要定时处理的场景都可以使用Spring Task

1.2 cron表达式

字符串形式,通过cron表达式可以定义任务触发的时间
构成规则:分为6或7个域,由空格分隔开,每个域代表一个含义
每个域的含义分别为:秒、分钟、小时、日、月、周、年(可选)
cron表达式在线生成器:https://cron.qqe2.com/
在这里插入图片描述

1.3 使用步骤

  1. 导入maven坐标 spring-context(已存在)
  2. 启动类添加注解 @EnableScheduling 开启任务调度
  3. 自定义定时任务类

二、订单状态定时处理

1. 业务规则

用户下单后可能存在的情况:

  1. 下单后未支付,订单一直处于“待支付”状态
  • 解决:
    通过定时任务每分钟检查一次是否存在支付超时订单(下单后超过15分钟仍未支付则判定为支付超时订单),如果存在则修改订单状态为“已取消”
  1. 用户收货后管理端未点击完成按钮,订单一直处于“派送中”状态
  • 解决:
    通过定时任务每天凌晨1点检查一次是否存在“派送中”的订单,如果存在则修改订单状态为“已完成”

2. 代码实现

1、自定义定时任务类OrderTask

@Component
@Slf4j
public class OrderTask {
    @Autowired
    private OrderMapper orderMapper;
   // 处理超时订单的方法
    @Scheduled(cron = "0 * * * * ? ") //每分钟触发一次
    //@Scheduled(cron = "1/5 * * * * ?")  //测试
    public void processTimeoutOrder(){
        log.info("定时处理超时订单:{}", LocalDateTime.now());
        //select * from orders where status = ? and orderTime < (当前时间-15分钟)
        List<Orders> ordersList = orderMapper.getByStatusAndOrderTimeT(Orders.PENDING_PAYMENT, LocalDateTime.now().plusMinutes(-15));
        if(ordersList!=null && ordersList.size()>0){
            for (Orders orders : ordersList) {
                orders.setStatus(Orders.CANCELLED);
                orders.setCancelReason("订单超时,自动取消");
                orders.setCancelTime(LocalDateTime.now());
                orderMapper.update(orders);
            }
        }
    }
    //处理一直处于派送中状态的订单
    @Scheduled(cron = "0 0 1 * * ? ")   //每天凌晨1点发一次
    //@Scheduled(cron = "0/5 * * * * ?")
    public void processDeliveryOrder(){
        log.info("定时处理处于派送中的订单:{}",LocalDateTime.now());
        List<Orders> ordersList = orderMapper.getByStatusAndOrderTimeT(Orders.DELIVERY_IN_PROGRESS, LocalDateTime.now().plusMinutes(-60));
        if(ordersList!=null && ordersList.size()>0){
            for (Orders orders : ordersList) {
                orders.setStatus(Orders.COMPLETED);
                orderMapper.update(orders);
            }
        }
    }
}

2、在OrderMapper接口中扩展方法,根据订单状态和下单时间查询订单

 @Select("select * from sky_take_out.orders where status = #{status} and order_time < #{orderTime}")
 List<Orders> getByStatusAndOrderTimeT(Integer status, LocalDateTime orderTime);

相关推荐

最近更新

  1. TCP协议是安全的吗?

    2024-04-23 07:08:05       16 阅读
  2. 阿里云服务器执行yum,一直下载docker-ce-stable失败

    2024-04-23 07:08:05       16 阅读
  3. 【Python教程】压缩PDF文件大小

    2024-04-23 07:08:05       15 阅读
  4. 通过文章id递归查询所有评论(xml)

    2024-04-23 07:08:05       18 阅读

热门阅读

  1. go语言学习

    2024-04-23 07:08:05       11 阅读
  2. Golang net/http 标准库源码学习

    2024-04-23 07:08:05       14 阅读
  3. 【嵌入式学习】ARM day04.16

    2024-04-23 07:08:05       12 阅读
  4. c++计算DNA探针的熔解温度

    2024-04-23 07:08:05       11 阅读
  5. MapReduce——数据切片与MapTask并行度决定机制

    2024-04-23 07:08:05       12 阅读
  6. 代码随想录:链表

    2024-04-23 07:08:05       14 阅读
  7. 分发糖果——使用贪心算法

    2024-04-23 07:08:05       13 阅读
  8. CentOS 7 上安装 MySQL 8.0详细步骤

    2024-04-23 07:08:05       12 阅读
  9. 前端需要知道的知识点,附有链接

    2024-04-23 07:08:05       15 阅读
  10. FPGA ——Verilog语法示例

    2024-04-23 07:08:05       16 阅读
  11. 【Leetcode】并查集/DFS/BFS多解

    2024-04-23 07:08:05       12 阅读