欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

SpringBoot中并發(fā)定時任務的實現(xiàn)、動態(tài)定時任務的實現(xiàn)(看這一篇就夠了)推薦

 更新時間:2019年04月06日 11:33:06   作者:會煉鋼的小白龍  
這篇文章主要介紹了SpringBoot并發(fā)定時任務動態(tài)定時任務實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧

一、在JAVA開發(fā)領域,目前可以通過以下幾種方式進行定時任務

1、單機部署模式

Timer:jdk中自帶的一個定時調(diào)度類,可以簡單的實現(xiàn)按某一頻度進行任務執(zhí)行。提供的功能比較單一,無法實現(xiàn)復雜的調(diào)度任務。
ScheduledExecutorService:也是jdk自帶的一個基于線程池設計的定時任務類。其每個調(diào)度任務都會分配到線程池中的一個線程執(zhí)行,所以其任務是并發(fā)執(zhí)行的,互不影響。
Spring Task:Spring提供的一個任務調(diào)度工具,支持注解和配置文件形式,支持Cron表達式,使用簡單但功能強大。
Quartz:一款功能強大的任務調(diào)度器,可以實現(xiàn)較為復雜的調(diào)度功能,如每月一號執(zhí)行、每天凌晨執(zhí)行、每周五執(zhí)行等等,還支持分布式調(diào)度,就是配置稍顯復雜。

2、分布式集群模式(不多介紹,簡單提一下)

問題:

  1. I、如何解決定時任務的多次執(zhí)行?
  2. II、如何解決任務的單點問題,實現(xiàn)任務的故障轉(zhuǎn)移?

問題I的簡單思考:

  1. 1、固定執(zhí)行定時任務的機器(可以有效避免多次執(zhí)行的情況 ,缺點就是單點故障問題)。
  2. 2、借助Redis的過期機制和分布式鎖。
  3. 3、借助mysql的鎖機制等。

成熟的解決方案:

1、Quartz:可以去看看這篇文章[Quartz分布式]( http://www.dbjr.com.cn/article/102869.htm)。
2、elastic-job:(https://github.com/elasticjob/elastic-job-lite)當當開發(fā)的彈性分布式任務調(diào)度系統(tǒng),采用zookeeper實現(xiàn)分布式協(xié)調(diào),實現(xiàn)任務高可用以及分片。
3、xxl-job:(https://github.com/xuxueli/xxl-job)是大眾點評員發(fā)布的分布式任務調(diào)度平臺,是一個輕量級分布式任務調(diào)度框架。
4、saturn:(https://github.com/vipshop/Saturn) 是唯品會提供一個分布式、容錯和高可用的作業(yè)調(diào)度服務框架。

二、SpringTask實現(xiàn)定時任務(這里是基于springboot)

1、簡單的定時任務實現(xiàn)

使用方式:

  1. 使用@EnableScheduling注解開啟對定時任務的支持。
  2. 使用@Scheduled 注解即可,基于corn、fixedRate、fixedDelay等一些定時策略來實現(xiàn)定時任務。

使用缺點:

  1. 1、多個定時任務使用的是同一個調(diào)度線程,所以任務是阻塞執(zhí)行的,執(zhí)行效率不高。
  2. 2、其次如果出現(xiàn)任務阻塞,導致一些場景的定時計算沒有實際意義,比如每天12點的一個計算任務被阻塞到1點去執(zhí)行,會導致結(jié)果并非我們想要的。

使用優(yōu)點:

  1. 1、配置簡單
  2. 2、適用于單個后臺線程執(zhí)行周期任務,并且保證順序一致執(zhí)行的場景   

 源碼分析:

//默認使用的調(diào)度器
if(this.taskScheduler == null) { 
 this.localExecutor = Executors.newSingleThreadScheduledExecutor();
 this.taskScheduler = new ConcurrentTaskScheduler(this.localExecutor);
}
//可以看到SingleThreadScheduledExecutor指定的核心線程為1,說白了就是單線程執(zhí)行
public static ScheduledExecutorService newSingleThreadScheduledExecutor() {
 return new DelegatedScheduledExecutorService
 (new ScheduledThreadPoolExecutor(1));
}
//利用了DelayedWorkQueue延時隊列作為任務的存放隊列,這樣便可以實現(xiàn)任務延遲執(zhí)行或者定時執(zhí)行
public ScheduledThreadPoolExecutor(int corePoolSize) {
 super(corePoolSize, Integer.MAX_VALUE, 0, NANOSECONDS,
  new DelayedWorkQueue());
}

2、實現(xiàn)并發(fā)的定時任務

使用方式:

方式一:由1中我們知道之所以定時任務是阻塞執(zhí)行,是配置的線程池決定的,那就好辦了,換一個不就行了!直接上代碼:

@Configuration
 public class ScheduledConfig implements SchedulingConfigurer {

 @Autowired
 private TaskScheduler myThreadPoolTaskScheduler;

 @Override
 public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
  //簡單粗暴的方式直接指定
  //scheduledTaskRegistrar.setScheduler(Executors.newScheduledThreadPool(5));
  //也可以自定義的線程池,方便線程的使用與維護,這里不多說了
  scheduledTaskRegistrar.setTaskScheduler(myThreadPoolTaskScheduler);
 }
 }

 @Bean(name = "myThreadPoolTaskScheduler")
 public TaskScheduler getMyThreadPoolTaskScheduler() {
 ThreadPoolTaskScheduler taskScheduler = new ThreadPoolTaskScheduler();
 taskScheduler.setPoolSize(10);
 taskScheduler.setThreadNamePrefix("Haina-Scheduled-");
 taskScheduler.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
 //調(diào)度器shutdown被調(diào)用時等待當前被調(diào)度的任務完成
 taskScheduler.setWaitForTasksToCompleteOnShutdown(true);
 //等待時長
 taskScheduler.setAwaitTerminationSeconds(60);
 return taskScheduler;
 } 

方式二:方式一的本質(zhì)改變了任務調(diào)度器默認使用的線程池,接下來這種是不改變調(diào)度器的默認線程池,而是把當前任務交給一個異步線程池去執(zhí)行

首先使用@EnableAsync 啟用異步任務
然后在定時任務的方法加上@Async即可,默認使用的線程池為SimpleAsyncTaskExecutor(該線程池默認來一個任務創(chuàng)建一個線程,就會不斷創(chuàng)建大量線程,極有可能壓爆服務器內(nèi)存。當然它有自己的限流機制,這里就不多說了,有興趣的自己翻翻源碼~)
項目中為了更好的控制線程的使用,我們可以自定義我們自己的線程池,使用方式@Async("myThreadPool")

廢話太多,直接上代碼:

 @Scheduled(fixedRate = 1000*10,initialDelay = 1000*20)
 @Async("myThreadPoolTaskExecutor")
 //@Async
 public void scheduledTest02(){
  System.out.println(Thread.currentThread().getName()+"--->xxxxx--->"+Thread.currentThread().getId());
 }

 //自定義線程池
 @Bean(name = "myThreadPoolTaskExecutor")
 public TaskExecutor getMyThreadPoolTaskExecutor() {
  ThreadPoolTaskExecutor taskExecutor = new ThreadPoolTaskExecutor();
  taskExecutor.setCorePoolSize(20);
  taskExecutor.setMaxPoolSize(200);
  taskExecutor.setQueueCapacity(25);
  taskExecutor.setKeepAliveSeconds(200);
  taskExecutor.setThreadNamePrefix("Haina-ThreadPool-");
  // 線程池對拒絕任務(無線程可用)的處理策略,目前只支持AbortPolicy、CallerRunsPolicy;默認為后者
  taskExecutor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
  //調(diào)度器shutdown被調(diào)用時等待當前被調(diào)度的任務完成
  taskExecutor.setWaitForTasksToCompleteOnShutdown(true);
  //等待時長
  taskExecutor.setAwaitTerminationSeconds(60);
  taskExecutor.initialize();
  return taskExecutor;
 }

線程池的使用心得(后續(xù)有專門文章來探討)

java中提供了ThreadPoolExecutor和ScheduledThreadPoolExecutor,對應與spring中的ThreadPoolTaskExecutor和ThreadPoolTaskScheduler,但是在原有的基礎上增加了新的特性,在spring環(huán)境下更容易使用和控制。
使用自定義的線程池能夠避免一些默認線程池造成的內(nèi)存溢出、阻塞等等問題,更貼合自己的服務特性
使用自定義的線程池便于對項目中線程的管理、維護以及監(jiān)控。
即便在非spring環(huán)境下也不要使用java默認提供的那幾種線程池,坑很多,阿里代碼規(guī)約不說了嗎,得相信大廠?。?!

三、動態(tài)定時任務的實現(xiàn)

問題:
使用@Scheduled注解來完成設置定時任務,但是有時候我們往往需要對周期性的時間的設置會做一些改變,或者要動態(tài)的啟停一個定時任務,那么這個時候使用此注解就不太方便了,原因在于這個注解中配置的cron表達式必須是常量,那么當我們修改定時參數(shù)的時候,就需要停止服務,重新部署。

解決辦法:

方式一:實現(xiàn)SchedulingConfigurer接口,重寫configureTasks方法,重新制定Trigger,核心方法就是addTriggerTask(Runnable task, Trigger trigger) ,不過需要注意的是,此種方式修改了配置值后,需要在下一次調(diào)度結(jié)束后,才會更新調(diào)度器,并不會在修改配置值時實時更新,實時更新需要在修改配置值時額外增加相關邏輯處理。

@Configuration
 public class ScheduledConfig implements SchedulingConfigurer {

 @Autowired
 private TaskScheduler myThreadPoolTaskScheduler;

 @Override
 public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
  //scheduledTaskRegistrar.setScheduler(Executors.newScheduledThreadPool(5));
  scheduledTaskRegistrar.setTaskScheduler(myThreadPoolTaskScheduler);
  //可以實現(xiàn)動態(tài)調(diào)整定時任務的執(zhí)行頻率
  scheduledTaskRegistrar.addTriggerTask(
    //1.添加任務內(nèi)容(Runnable)
    () -> System.out.println("cccccccccccccccc--->" + Thread.currentThread().getId()),
    //2.設置執(zhí)行周期(Trigger)
    triggerContext -> {
     //2.1 從數(shù)據(jù)庫動態(tài)獲取執(zhí)行周期
     String cron = "0/2 * * * * ? ";
     //2.2 合法性校驗.
 //     if (StringUtils.isEmpty(cron)) {
 //      // Omitted Code ..
 //     }
      //2.3 返回執(zhí)行周期(Date)
      return new CronTrigger(cron).nextExecutionTime(triggerContext);
     }
   );
 }
 }

方式二:使用threadPoolTaskScheduler類可實現(xiàn)動態(tài)添加刪除功能,當然也可實現(xiàn)執(zhí)行頻率的調(diào)整

首先,我們要認識下這個調(diào)度類,它其實是對java中ScheduledThreadPoolExecutor的一個封裝改進后的產(chǎn)物,主要改進有以下幾點:

  1. 1、提供默認配置,因為是ScheduledThreadPoolExecutor,所以只有poolSize這一個默認參數(shù)。
  2. 2、支持自定義任務,通過傳入Trigger參數(shù)。
  3. 3、對任務出錯處理進行優(yōu)化,如果是重復性的任務,不拋出異常,通過日志記錄下來,不影響下次運行,如果是只執(zhí)行一次的任務,將異常往上拋。

順便說下ThreadPoolTaskExecutor相對于ThreadPoolExecutor的改進點:

  1. 1、提供默認配置,原生的ThreadPoolExecutor的除了ThreadFactory和RejectedExecutionHandler其他沒有默認配置
  2. 2、實現(xiàn)AsyncListenableTaskExecutor接口,支持對FutureTask添加success和fail的回調(diào),任務成功或失敗的時候回執(zhí)行對應回調(diào)方法。
  3. 3、因為是spring的工具類,所以拋出的RejectedExecutionException也會被轉(zhuǎn)換為spring框架的TaskRejectedException異常(這個無所謂)
  4. 4、提供默認ThreadFactory實現(xiàn),直接通過參數(shù)重載配置

扯了這么多,還是直接上代碼:

 @Component
 public class DynamicTimedTask {

  private static final Logger logger = LoggerFactory.getLogger(DynamicTimedTask.class);

  //利用創(chuàng)建好的調(diào)度類統(tǒng)一管理
  //@Autowired
  //@Qualifier("myThreadPoolTaskScheduler")
  //private ThreadPoolTaskScheduler myThreadPoolTaskScheduler;


  //接受任務的返回結(jié)果
  private ScheduledFuture<?> future;

  @Autowired
  private ThreadPoolTaskScheduler threadPoolTaskScheduler;

  //實例化一個線程池任務調(diào)度類,可以使用自定義的ThreadPoolTaskScheduler
  @Bean
  public ThreadPoolTaskScheduler threadPoolTaskScheduler() {
   ThreadPoolTaskScheduler executor = new ThreadPoolTaskScheduler();
   return new ThreadPoolTaskScheduler();
  }


  /**
  * 啟動定時任務
  * @return
  */
  public boolean startCron() {
   boolean flag = false;
   //從數(shù)據(jù)庫動態(tài)獲取執(zhí)行周期
   String cron = "0/2 * * * * ? ";
   future = threadPoolTaskScheduler.schedule(new CheckModelFile(),cron);
   if (future!=null){
    flag = true;
    logger.info("定時check訓練模型文件,任務啟動成功?。?!");
   }else {
    logger.info("定時check訓練模型文件,任務啟動失?。。?!");
   }
   return flag;
  }

  /**
  * 停止定時任務
  * @return
  */
  public boolean stopCron() {
   boolean flag = false;
   if (future != null) {
    boolean cancel = future.cancel(true);
    if (cancel){
     flag = true;
     logger.info("定時check訓練模型文件,任務停止成功!??!");
    }else {
     logger.info("定時check訓練模型文件,任務停止失?。。?!");
    }
   }else {
    flag = true;
    logger.info("定時check訓練模型文件,任務已經(jīng)停止!??!");
   }
   return flag;
  }


  class CheckModelFile implements Runnable{

   @Override
   public void run() {
    //編寫你自己的業(yè)務邏輯 
    System.out.print("模型文件檢查完畢!??!")
   }
  }

 }

四、總結(jié)

到此基于springtask下的定時任務的簡單使用算是差不多了,其中不免有些錯誤的地方,或者理解有偏頗的地方歡迎大家提出來!
基于分布式集群下的定時任務使用,后續(xù)有時間再繼續(xù)?。?!

以上所述是小編給大家介紹的SpringBoot并發(fā)定時任務動態(tài)定時任務實現(xiàn)詳解整合,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!

相關文章

  • Java 如何優(yōu)雅的拷貝對象屬性

    Java 如何優(yōu)雅的拷貝對象屬性

    這篇文章主要介紹了Java 如何優(yōu)雅的拷貝對象屬性,幫助大家更好的理解和學習Java,感興趣的朋友可以了解下
    2020-11-11
  • SpringSecurity攔截器鏈的使用詳解

    SpringSecurity攔截器鏈的使用詳解

    這篇文章主要介紹了SpringSecurity攔截器鏈的使用詳解,webSecurity的build方法最終調(diào)用的是doBuild方法,doBuild方法調(diào)用的是webSecurity的performBuild方法,webSecurity完成所有過濾器的插件,最終返回的是過濾器鏈代理類filterChainProxy,需要的朋友可以參考下
    2023-11-11
  • java中同類對象之間的compareTo()和compare()方法對比分析

    java中同類對象之間的compareTo()和compare()方法對比分析

    這篇文章主要介紹了java中同類對象之間的compareTo()和compare()方法對比分析,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-09-09
  • springboot+redis+阿里云短信實現(xiàn)手機號登錄功能

    springboot+redis+阿里云短信實現(xiàn)手機號登錄功能

    這篇文章主要介紹了springboot+redis+阿里云短信實現(xiàn)手機號登錄功能,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2024-01-01
  • java實現(xiàn)簡單控制臺通訊錄

    java實現(xiàn)簡單控制臺通訊錄

    這篇文章主要為大家詳細介紹了java實現(xiàn)簡單控制臺通訊錄,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-02-02
  • 解決spirngboot連接redis報錯:READONLY?You?can‘t?write?against?a?read?only?replica的問題

    解決spirngboot連接redis報錯:READONLY?You?can‘t?write?against?

    docker部署的redis,springboot基本每天來連redis都報錯:READONLY?You?can't?write?against?a?read?only?replica,重啟redis后,可以正常連接。但是每天都重啟redis,不現(xiàn)實,也很麻煩,今天給大家分享解決方式,感興趣的朋友一起看看吧
    2023-06-06
  • spring?cloud?gateway中netty線程池小優(yōu)化

    spring?cloud?gateway中netty線程池小優(yōu)化

    這篇文章主要介紹了spring?cloud?gateway中netty線程池小優(yōu)化技巧示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-07-07
  • Springboot實現(xiàn)Shiro整合JWT的示例代碼

    Springboot實現(xiàn)Shiro整合JWT的示例代碼

    這篇文章主要介紹了Springboot實現(xiàn)Shiro整合JWT的示例代碼,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2020-12-12
  • 深入了解Java.Util.Date詳情

    深入了解Java.Util.Date詳情

    這篇文章主要介紹了Java.Util.Date,很少有類能像java.util.Date那樣在堆棧溢出方面引起如此多的類似問題,關于具體原因下文內(nèi)容詳細介紹,需要的朋友可以參考一下
    2022-06-06
  • 在spring-boot工程中添加spring mvc攔截器

    在spring-boot工程中添加spring mvc攔截器

    這篇文章主要介紹了在spring-boot工程中添加spring mvc攔截器,Spring MVC的攔截器(Interceptor)不是Filter,同樣可以實現(xiàn)請求的預處理、后處理。,需要的朋友可以參考下
    2019-06-06

最新評論