Spring Task用于在指定时间处理某个业务逻辑,在本项目中体现在订单状态定时处理,包括“下单后未支付(订单超时自动取消)”、“订单一直处于派送中状态(订单需要自动更新成已完成)”。
实现的业务逻辑通过当前订单状态和下单时间去查询订单,对于满足上面两个条件的订单,设置一个时间让任务自动执行。
Spring框架提供的任务调度工具,可以按照约定的时间自动执行某个代码逻辑。
应用场景
字符串形式,通过cron表达式可以定义任务触发的时间
构成规则:分为6或7个域,由空格分隔开,每个域代表一个含义
每个域的含义分别为:秒、分钟、小时、日、月、周、年(可选)
cron表达式在线生成器:https://cron.qqe2.com/
用户下单后可能存在的情况:
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);