1:Spring Task概念:
Spring Task 是Spring框架提供的任务调度工具,可以按照约定的时间自动执行某个代码逻辑
定时任务的理解
定时任务即系统在特定时间执行一段代码,它的场景应用非常广泛:
- 购买游戏的月卡会员后,系统每天给会员发放游戏资源。
- 管理系统定时生成报表。
- 定时清理系统垃圾。
Cron表达式
cron表达式其实就是一个字符串,通过cron表达式可以定义任务触发的时间
构成规则:分为6或7个域,由空格分隔开,每个域代表一个含义
每个域的含义分别为:秒、分钟、小时、日、月、周、年(可选)
cron表达式在线生成器:https://cron.qqe2.com/
2:Spring Task入门案例:
Spring Task使用步骤:
- 导入maven坐标 spring-context(已存在)
- 启动类添加注解 @EnableScheduling
- 开启任务调度 自定义定时任务类
@Component
@Slf4j
public class MyTask {
@Scheduled(cron = "0/5 * * * * ? ")
public void ScheduledTest(){
log.info("Spring Task执行:{}",new Date());
}
}
3:订单状态定时处理:
需求分析:
用户下单后可能存在的情况:
- 下单后未支付,订单一直处于“待支付”状态
- 用户收货后管理端未点击完成按钮,订单一直处于“派送中”状态
解决方法:
对于上面两种情况需要通过定时任务来修改订单状态,
具体逻辑为: 通过定时任务每分钟检查一次是否存在支付超时订单(下单后超过15分钟仍未支付则判定为支付超时订单),如果存在则修改订单状态为“已取消”
通过定时任务每天凌晨1点检查一次是否存在“派送中”的订单,如果存在则修改订单状态为“已完成”
具体代码实现:
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.core.annotation.Order;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;
import java.time.LocalDateTime;
import java.util.List;
/*
订单状态处理
*/
@Component
@Slf4j
public class OrderTask {
@Autowired
private OrderMapper orderMapper;
/**
* 处理超时订单
*/
@Scheduled(cron = "0 * * * * ? ")//每分钟
public void dealTimeOutOrder(){
log.info("处理超时订单");
LocalDateTime time = LocalDateTime.now().plusMinutes(-15);
//select * from orders where status 1 and orderTime <(now - 15)
List<Orders> orderList = orderMapper.selectTimeOutOrder(Orders.PENDING_PAYMENT,time);
if(!orderList.isEmpty()){
for (Orders orders : orderList) {
orders.setStatus(Orders.CANCELLED);
orders.setCancelReason("订单超时,自动取消");
orders.setCancelTime(LocalDateTime.now());
orderMapper.update(orders);
}
}
}
/*
处理一直处于在派送中的订单
*/
@Scheduled(cron = "0 0 1 * * ?")//每天凌晨一点
public void dealProcessingOrder(){
log.info("处理一直处于在派送中的订单");
LocalDateTime time = LocalDateTime.now().plusMinutes(-60);//上一个工作日减去60mins
final List<Orders> ordersList = orderMapper.selectTimeOutOrder(Orders.DELIVERY_IN_PROGRESS, time);
if(!ordersList.isEmpty()){
for (Orders orders : ordersList) {
orders.setStatus(Orders.COMPLETED);
orderMapper.update(orders);
}
}
}
}
/**
* 处理超时订单
*/
@Select("select * from sky_take_out.orders where status = #{pendingPayment} and order_time < #{time}")
List<Orders> selectTimeOutOrder(Integer pendingPayment, LocalDateTime time);
两个问题处理的逻辑其实差不多,就是发sql查询数据库中是否有不符合要求的订单
有的话就更新订单就行。