1酌畜、使用注解方式
? 首先需要在啟動類下添加 @EnableScheduling 注解(@EnableAsync是開啟異步的注解)
package com.fongtech.cli;
import org.mybatis.spring.annotation.MapperScan;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.scheduling.annotation.EnableAsync;
import org.springframework.scheduling.annotation.EnableScheduling;
@SpringBootApplication
@MapperScan("com.fongtech.cli.mbg.*.**")
@EnableAsync
@EnableScheduling
public class SpringbootAdminApplication {
? ? public static void main(String[] args) {
? ? ? ? SpringApplication.run(SpringbootAdminApplication.class, args);
? ? }
}
? 接著在需要用到定時任務的類和方法下加 @Component 和 @Scheduled(cron = "0 0/1 * * * ? ")注解,其中@Scheduled()中的 ‘cron’ 有固定的格式。(@Async注解表示開啟異步)
@Slf4j
@Component
public class AsyncTaskConfiguration {
? ? /**
? ? * 每分鐘檢查任務列表织鲸,判斷任務類型執(zhí)行相應的任務
? ? * 根據實際任務執(zhí)行情況,限定執(zhí)行任務數量
? ? */
? ? @Scheduled(cron = "0 0/1 * * * ? ")
? ? @Async
? ? public void startCommonTask() throws Exception {
? ? ? ? log.info("startCommonTask? start........." + Thread.currentThread().getName());
? ? ? ? commonTaskService.startCommonTask();
? ? ? ? log.info("startCommonTask? end........." + Thread.currentThread().getName());
? ? }}
2棍丐、使用實現接口的方式
? 通過實現 SchedulingConfigurer 接口渠牲,可對定時任務進行操作。實現接口的方式相比使用注解更加靈活泌参,但需要編寫代碼脆淹,相對繁瑣。
? 實現工具類如下:
package com.fongtech.cli.admin.tasktime;
import com.fongtech.cli.common.util.BeanUtils;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.SchedulingException;
import org.springframework.scheduling.TaskScheduler;
import org.springframework.scheduling.annotation.EnableScheduling;
import org.springframework.scheduling.annotation.SchedulingConfigurer;
import org.springframework.scheduling.config.ScheduledTaskRegistrar;
import org.springframework.scheduling.config.TriggerTask;
import org.springframework.scheduling.support.CronTrigger;
import javax.annotation.PostConstruct;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ScheduledFuture;
/**
* @author linb
* @date 2020/6/15 11:16
*/
@Configuration
//@EnableScheduling
public class DefaultSchedulingConfigurer implements SchedulingConfigurer {
? ? private ScheduledTaskRegistrar taskRegistrar;
? ? private Set<ScheduledFuture<?>> scheduledFutures = null;
? ? private Map<String, ScheduledFuture<?>> taskFutures = new ConcurrentHashMap<>();
? ? @Override
? ? public void configureTasks(ScheduledTaskRegistrar taskRegistrar) {
? ? ? ? this.taskRegistrar = taskRegistrar;
? ? }
? ? @SuppressWarnings("unchecked")
? ? private Set<ScheduledFuture<?>> getScheduledFutures() {
? ? ? ? if (scheduledFutures == null) {
? ? ? ? ? ? try {
? ? ? ? ? ? ? ? // spring版本不同選用不同字段scheduledFutures
? ? ? ? ? ? ? ? scheduledFutures = (Set<ScheduledFuture<?>>) BeanUtils.getProperty(taskRegistrar, "scheduledTasks");
? ? ? ? ? ? } catch (NoSuchFieldException e) {
? ? ? ? ? ? ? ? throw new SchedulingException("not found scheduledFutures field.");
? ? ? ? ? ? }
? ? ? ? }
? ? ? ? return scheduledFutures;
? ? }
? ? /**
? ? * 添加任務
? ? */
? ? public void addTriggerTask(String taskId, TriggerTask triggerTask) {
? ? ? ? if (taskFutures.containsKey(taskId)) {
? ? ? ? ? ? throw new SchedulingException("the taskId[" + taskId + "] was added.");
? ? ? ? }
? ? ? ? TaskScheduler scheduler = taskRegistrar.getScheduler();
? ? ? ? ScheduledFuture<?> future = scheduler.schedule(triggerTask.getRunnable(), triggerTask.getTrigger());
? ? ? ? getScheduledFutures().add(future);
? ? ? ? taskFutures.put(taskId, future);
? ? }
? ? /**
? ? * 取消任務
? ? */
? ? public void cancelTriggerTask(String taskId) {
? ? ? ? ScheduledFuture<?> future = taskFutures.get(taskId);
? ? ? ? if (future != null) {
? ? ? ? ? ? future.cancel(true);
? ? ? ? }
? ? ? ? taskFutures.remove(taskId);
? ? ? ? getScheduledFutures().remove(future);
? ? }
? ? /**
? ? * 重置任務
? ? */
? ? public void resetTriggerTask(String taskId, TriggerTask triggerTask) {
? ? ? ? cancelTriggerTask(taskId);
? ? ? ? addTriggerTask(taskId, triggerTask);
? ? }
? ? /**
? ? * 任務編號
? ? */
? ? public Set<String> taskIds() {
? ? ? ? return taskFutures.keySet();
? ? }
? ? /**
? ? * 是否存在任務
? ? */
? ? public boolean hasTask(String taskId) {
? ? ? ? return this.taskFutures.containsKey(taskId);
? ? }
? ? /**
? ? * 任務調度是否已經初始化完成
? ? */
? ? public boolean inited() {
? ? ? ? return this.taskRegistrar != null && this.taskRegistrar.getScheduler() != null;
? ? }
}
? 在項目啟動后就自動開啟任務的操作類如下:
package com.fongtech.cli.admin.tasktime;
import com.fongtech.cli.admin.service.IAuthLoginService;
import com.fongtech.cli.admin.service.IBackupsService;
import com.fongtech.cli.admin.service.IDictionnaryEntryService;
import com.fongtech.cli.mbg.model.entity.AuthLogin;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.CommandLineRunner;
import org.springframework.core.annotation.Order;
import org.springframework.scheduling.config.TriggerTask;
import org.springframework.scheduling.support.CronTrigger;
import org.springframework.stereotype.Component;
/**
* 項目啟動后執(zhí)行沽一,
*/
@Slf4j
@Component
@Order(value = 1)
public class CmdRunner implements CommandLineRunner {
? ? @Autowired
? ? private DefaultSchedulingConfigurer defaultSchedulingConfigurer;
? ? @Autowired
? ? private IDictionnaryEntryService dictionnaryEntryService;
? ? @Autowired
? ? private IBackupsService backupsService;
? ? @Autowired
? ? private IAuthLoginService authLoginService;
? ? @Override
? ? public void run(String... args) throws Exception {
? ? ? ? log.info("------按照預設備份周期啟動數據庫備份定時任務");
? ? ? ? while (!defaultSchedulingConfigurer.inited())
? ? ? ? {
? ? ? ? ? ? try {
? ? ? ? ? ? ? ? Thread.sleep(100);
? ? ? ? ? ? } catch (InterruptedException e) {
? ? ? ? ? ? ? ? e.printStackTrace();
? ? ? ? ? ? } finally {
? ? ? ? ? ? }
? ? ? ? }
? ? ? ? String cron = dictionnaryEntryService.getEntryValueByName("CRON_VALUE");
? ? ? ? //默認按照管理員用戶權限執(zhí)行備份任務
? ? ? ? AuthLogin authLogin = authLoginService.query().eq(AuthLogin::getLogin_user, "admin").getOne();
? ? ? ? //啟動線程盖溺,按照原表內的時間執(zhí)行備份任務
? ? ? ? defaultSchedulingConfigurer.addTriggerTask("task",
? ? ? ? ? ? ? ? new TriggerTask(
? ? ? ? ? ? ? ? ? ? ? ? () -> System.out.println("=====----------啟動定時任務=-----------");,
? ? ? ? ? ? ? ? ? ? ? ? new CronTrigger(cron)));
? ? }
}
? 暫停定時任務:
defaultSchedulingConfigurer.cancelTriggerTask("task");