前言
Spring Task用于在指定时间处理某个业务逻辑,在本项目中体现在订单状态定时处理,包括“下单后未支付(订单超时自动取消)”、“订单一直处于派送中状态(订单需要自动更新成已完成)”。
实现的业务逻辑通过当前订单状态和下单时间去查询订单,对于满足上面两个条件的订单,设置一个时间让任务自动执行。
一、Spring Task
1.1 概述
Spring框架提供的任务调度工具,可以按照约定的时间自动执行某个代码逻辑。
应用场景
- 信用卡每月还款提醒
- 银行贷款每月还款提醒
- 火车票售票系统处理未支付订单
- 入职纪念日为用户发送通知
只要是需要定时处理的场景都可以使用Spring Task
1.2 cron表达式
字符串形式,通过cron表达式可以定义任务触发的时间
构成规则:分为6或7个域,由空格分隔开,每个域代表一个含义
每个域的含义分别为:秒、分钟、小时、日、月、周、年(可选)
cron表达式在线生成器:https://cron.qqe2.com/
1.3 使用步骤
- 导入maven坐标 spring-context(已存在)
- 启动类添加注解 @EnableScheduling 开启任务调度
- 自定义定时任务类
二、订单状态定时处理
1. 业务规则
用户下单后可能存在的情况:
- 下单后未支付,订单一直处于“待支付”状态
- 解决:
通过定时任务每分钟检查一次是否存在支付超时订单(下单后超过15分钟仍未支付则判定为支付超时订单),如果存在则修改订单状态为“已取消”
- 用户收货后管理端未点击完成按钮,订单一直处于“派送中”状态
- 解决:
通过定时任务每天凌晨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);