介绍
Spring Task是Spring框架提供的任务调度工具,可以按照约定的时间自动执行某个代码逻辑。
定位: 定时任务框架
作用: 定时自动执行某段Java代码
应用场景: 引用卡每月还款提醒、银行贷款每月还款提醒、火车票售票系统处理未支付订单……
cron表达式
cron表达式其实就是一个字符串,通过cron表达式可以定义任务触发的时间
构成规则:分为6或7个域,由空格分隔开,每个域代表一个含义
每个域的含义分别为:秒、分钟、小时、日、月、周(表示星期几)、年(可选)
举例:
注意: 表达式中的周和日往往不能同时出现(避免出错),一般指定其中一个,另一个用“?”代替。
cron表达式在线生成器:https://cron.qqe2.com/
案例
Spring Task使用步骤:
- 导入maven坐标 spring-context
- 在启动类添加注解 ——通过@EnableScheduling 开启任务调度
- 自定义定时任务类(规定任务执行的时间和代码)
自定义定时任务类举例:
import lombok.extern.slf4j.Slf4j;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;
import java.util.Date;
@Component // 表示该类需要实例化,并且交给spring容器管理
@Slf4j
public class MyTask {
/**
* 定时任务需要执行的业务
*/
@Scheduled(cron = "0/5 * * * * ?") // 每隔5秒执行一次
public void executeTask() {
log.info("定时任务开始执行:{}", new Date());
}
}
应用
场景:用户下单后可能出现的情况:
- 下单后订单未支付,订单一直处于“待支付”状态;
- 用户收货后,管理端未点击完成按钮,订单一直处于“派送中”状态
对于上面的情况,需要用过定时任务来修改订单状态,具体逻辑为:
- 通过定时任务每分钟检查一次是否存在支付超时订单(下单后超过15分钟仍未支付则判定为支付超时订单),如果存在则修改订单状态为“已取消”;
- 通过定时任务每天凌晨1点检查一次(打烊后)是否存在派送中的订单,如果存在则修改订单状态为已完成
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
@Slf4j
public class OrderTask {
@Autowired
private OrderMapper orderMapper;
/**
* 处理超时订单方法
*/
@Scheduled(cron = "0 * * * * ? ")// 每分钟触发一次
public void processTimeoutOrder(){
log.info("定时处理超时订单:{}", LocalDateTime.now());
// 查询超时订单
// select * from orders where status = ? and order_time < (当前时间 - 15min)
LocalDateTime time = LocalDateTime.now().plusMinutes(-15);
List<Orders> orderList = orderMapper.getByStatusAndOrderTimeLT(Orders.PENDING_PAYMENT, time);
if(orderList != null && orderList.size() > 0){
for (Orders orders : orderList) {
orders.setStatus(Orders.CANCELLED);
orders.setCancelReason("订单超时,自动取消");
orders.setCancelTime(LocalDateTime.now());
orderMapper.update(orders);
}
}
}
/**
* 处理一直处于派送中的订单
*/
@Scheduled(cron = "0 0 1 * * ? *")
public void processDeliveryOrder(){
log.info("定时处理处于派送中的订单:{}",LocalDateTime.now());
List<Orders> ordersList = orderMapper.getByStatusAndOrderTimeLT(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);
}
}
}
}