1.使用@Scheduled注解实现
@AllArgsConstructor
@Slf4j
@Component
public class ScheduledTasks {
//此处可以注入其他服务
/**
"0/5 * * * * ?" 每5秒触发
"0 0/2 * * * ?" 每2分钟执行一次
"0 0 12 * * ?" 每天中午十二点触发
"0 15 10 ? * *" 每天早上10:15触发
"0 15 10 * * ?" 每天早上10:15触发
"0 15 10 * * ? *" 每天早上10:15触发
"0 15 10 * * ? 2005" 2005年的每天早上10:15触发
"0 * 14 * * ?" 每天从下午2点开始到2点59分每分钟一次触发
"0 0/5 14 * * ?" 每天从下午2点开始到2:55分结束每5分钟一次触发
"0 0/5 14,18 * * ?" 每天的下午2点至2:55和6点至6点55分两个时间段内每5分钟一次触发
"0 0-5 14 * * ?" 每天14:00至14:05每分钟一次触发
"0 10,44 14 ? 3 WED" 三月的每周三的14:10和14:44触发
"0 15 10 ? * MON-FRI" 每个周一、周二、周三、周四、周五的10:15触发
*/
@Scheduled(cron="0 55 23 ? * *") //每天晚上23点55分触发执行一次
public void startTask(){
log.info("任务在执行...");
}
2.利用线程池ScheduledExecutorService
@Component
@Slf4j
@Order(1)
public class SoftGatewayHeartbeat implements ApplicationRunner {
//此处可以注入其他服务
@Override
public void run(ApplicationArguments args) throws Exception {
ScheduledExecutorService executorService = Executors.newScheduledThreadPool(1);
executorService.scheduleAtFixedRate(new Runnable() {
@Override
public void run() {
log.info("任务在执行...");
}
},2,cycle, TimeUnit.SECONDS);
}
}
利用线程池做定时任务有个问题,就是任务启动时机的问题,案例采用了线程池跟随系统一起启动,类实现了ApplicationRunner接口的run方法,注解的@Order(1)主要用来对多个随系统一起启动的任务进行排序。
一般这种适用于设计一些心跳场景的设计和实现,如虚拟设备上线后,给服务器或主机上报心跳信息。