文章目录
-
- 概要
- 整体架构流程
- 技术细节
- 小结
概要
订单定时状态处理通常涉及到对订单状态进行定期检查,并根据订单的状态自动执行某些操作,比如关闭未支付的订单、自动确认收货等.
需求分析以及接口设计
需求分析
用户下单后可能存在的情况:
-
下单后15min内未支付,订单一直处于“待支付”状态
-
用户收货后管理端未点击完成按钮,订单一直处于“派送中”状态
技术细节
对于上面两种情况需要通过定时任务来修改订单状态,具体逻辑为:
-
通过定时任务每分钟检查一次是否存在支付超时订单(下单后超过15分钟仍未支付则判定为支付超时订单),如果存在则修改订单状态为“已取消”
-
通过定时任务每天凌晨1点检查一次是否存在“派送中”的订单,如果存在则修改订单状态为“已完成”
1.task:
由于Springtask定时状态处理不需要通过接口调用,是自动执行的,所以不需要controller层
package com.sky.task;import com.sky.entity.Orders;
import com.sky.mapper.OrderMapper;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;import java.time.LocalDateTime;
import java.util.List;@Component//加入Spring容器
@Slf4j
public class OrderTask {@Autowiredprivate OrderMapper orderMapper;@Scheduled(cron = "0 * * * * ?")//每分钟执行一次public void processTimeOutOrder(){log.info("定时处理订单超时的订单");LocalDateTime orderTime = LocalDateTime.now().plusMinutes(-15);//查询到未支付且超时的订单List<Orders> ordersList = orderMapper.getByStatusAndOrderTimeLT(Orders.UN_PAID,orderTime);//遍历这些订单,更改他们的状态,取消时间和取消原因,再更新表if(ordersList != null && ordersList.size() > 0){for (Orders orders : ordersList) {orders.setStatus(Orders.CANCELLED);orders.setCancelTime(LocalDateTime.now());orders.setCancelReason("订单超时,已被自动取消");orderMapper.update(orders);}}}@Scheduled(cron = "0 0 1 * * ? ")//每天凌晨一点执行一次public void processDeliveryOrder(){log.info("定时处理派送中的订单");LocalDateTime orderTime = LocalDateTime.now().plusMinutes(60);//每天凌晨0点执行一次//查询到正在派送的订单List<Orders> ordersList = orderMapper.getByStatusAndOrderTimeLT(Orders.DELIVERY_IN_PROGRESS,orderTime);//遍历这些订单,更改他们的状态,取消时间和取消原因,再更新表if(ordersList != null && ordersList.size() > 0){for (Orders orders : ordersList) {orders.setStatus(Orders.COMPLETED);orderMapper.update(orders);}}}}
2.Mapper层:
@Select("select * from `sky-take-out`.orders where status = #{status} and order_time < #{orderTime}")List<Orders> getByStatusAndOrderTimeLT(Integer status, LocalDateTime orderTime);