1.定时任务适用场景
在日常生产的时候难免遇见这种情况,需要定时任务去处理,像信用卡的还款消息,订单未支付取消支付处理,员工入职周年庆等等场景。下面我将以订单超时未处理为例,进行讲解。
2.corn表达式
在使用SpringTask时需要先了解一下其中最重要的一点corn表达式。
2.1.corn表达式初步认识
cron表达式其实就是一个字符串,通过cron表达式可以定义任务触发的时间
构成规则:分为6或7个域,由空格分隔开,每个域代表一个含义
每个域的含义分别为:秒、分钟、小时、日、月、周、年(可选)
举例:
2022年10月12日上午9点整 对应的cron表达式为:0 0 9 12 10 ? 2022
说明:一般日和周的值不同时设置,其中一个设置,另一个用?表示。
比如:描述2月份的最后一天,最后一天具体是几号呢?可能是28号,也有可能是29号,所以就不能写具体数字。
为了描述这些信息,提供一些特殊的字符。这些具体的细节,我们就不用自己去手写,因为这个cron表达式,它其实有在线生成器。
cron表达式在线生成器:在线Cron表达式生成器
2.2.通配符
通配符:
* 表示所有值;
? 表示未说明的值,即不关心它为何值;
- 表示一个指定的范围;
, 表示附加一个可能值;
/ 符号前表示开始时间,符号后表示每次递增的值;
2.3示例
cron表达式案例:
*/5 * * * * ? 每隔5秒执行一次
0 */1 * * * ? 每隔1分钟执行一次
0 0 5-15 * * ? 每天5-15点整点触发
0 0/3 * * * ? 每三分钟触发一次
0 0-5 14 * * ? 在每天下午2点到下午2:05期间的每1分钟触发
0 0/5 14 * * ? 在每天下午2点到下午2:55期间的每5分钟触发
0 0/5 14,18 * * ? 在每天下午2点到2:55期间和下午6点到6:55期间的每5分钟触发
0 0/30 9-17 * * ? 朝九晚五工作时间内每半小时
0 0 10,14,16 * * ? 每天上午10点,下午2点,4点
3.SpringTasK的使用
亘古不变,首先导入依赖包,
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
</dependency>
然后在启动类上添加@EnableScheduling注解
然后编写定时任务类,corn表达式并不是唯一指定什么时候运行的方法,但是是最常用的,还有以下方法
-
fixRate: 固定频率
-
fixDelay:固定延迟
@Component
@Slf4j
public class MyTask {
/**
* 定时任务每五秒执行一次
*/
@Scheduled(cron = "0/5 * * * * ?")
public void task(){
log.info(LocalDateTime.now().toString());
}
}
切记!一定要放到ioc容器中去,不放进去是扫不到的。可能萌新不知道什么是ioc,其是spring的两大特性之一,在这不多做赘述,可以参考我这篇文章。
+++++++++++++++++++++++++++++++++++++++++++++++++++++++++
4.场景带入
订单超时未处理的场景我相信只要是在淘宝买过东西的都有遇见过吧,那么为什么要取消呢,这是因为一般订单在创建的时候就会进行库存预扣,避免长时间锁单,所以要定时清除未支付的订单。
下面我用简单的代码进行阐述。
简单的建表语句
create table `order`
(
order_id int auto_increment,
status int null comment '0:未支付;1:已支付',
crTime datetime null comment '创建时间',
chTime datetime null comment '修改时间',
constraint order_pk
primary key (order_id)
);
然后就是代码实现,因为service层跟mapper层都知道,为了方便阅读不再呈现,实在不知道可以留言
@Autowired
OrderService orderService;
@Scheduled(cron = "0/5 * * * * ?")
public void task(){
//查询到状态为未支付的订单
List<Order> orders = orderService.selectBystatus();
//获得到现在位置最晚的超时时间,可以将超时时间缩短,方便测试
LocalDateTime time = LocalDateTime.now().minusMinutes(15);
//过滤出当前已经超时的订单
List<Order> collect = orders.stream()
.filter(order -> order.getCrTime().isBefore(time))
.collect(Collectors.toList());
//将超时的订单进行状态修改
orderService.updateStatus(collect);
}
这就可以将超时未支付的订单进行取消处理了,当然这种处理方法肯定是不妥善的,因为是超时取消,就比较实时,定时频率就要足够快那么在高频率的定时扫库势必会对数据库造成影响。
那么有没有什么好办法,当然是有的。那就是延迟消息进行取消。感兴趣可以继续阅读我这篇文章。
+++++++++++++++++++++++++++++++++++++