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 {@Autowiredprivate 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);//上一个工作日减去60minsfinal 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查询数据库中是否有不符合要求的订单
有的话就更新订单就行。