• 苍穹外卖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);
                }
            }
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35

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

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

  • 相关阅读:
    Kafka 可视化工具
    4.4 审计
    字节跳动数据平台技术揭秘:基于 ClickHouse 的复杂查询实现与优化
    期货十三篇 第三篇 计划篇
    【Servlet】3:Servlet 的基本原理、Servlet对象的生命周期
    [MRCTF2020]套娃
    10、JAVA入门——多重循环及程序调试
    基于libopenh264 codec的svc分层流实现方案
    【Python】保姆级万字讲解:Python中的 pip 和 conda 的理解
    C#实现图的深度优先遍历递归算法--详细代码
  • 原文地址:https://blog.csdn.net/qq_51076441/article/details/138094191