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

java 定時器線程池(ScheduledThreadPoolExecutor)的實現(xiàn)

 更新時間:2020年06月18日 08:35:40   作者:Java專職  
這篇文章主要介紹了java 定時器線程池(ScheduledThreadPoolExecutor),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧

前言

定時器線程池提供了定時執(zhí)行任務的能力,即可以延遲執(zhí)行,可以周期性執(zhí)行。但定時器線程池也還是線程池,最底層實現(xiàn)還是ThreadPoolExecutor,可以參考我的另外一篇文章多線程–精通ThreadPoolExecutor。

特點說明

1.構造函數(shù)

 public ScheduledThreadPoolExecutor(int corePoolSize) {
 // 對于其他幾個參數(shù)在ThreadPoolExecutor中都已經詳細分析過了,所以這里,將不再展開
 // 這里我們可以看到調用基類中的方法時有個特殊的入參DelayedWorkQueue。
 // 同時我們也可以發(fā)現(xiàn)這里并沒有設置延遲時間、周期等參數(shù)入口。
 // 所以定時執(zhí)行的實現(xiàn)必然在DelayedWorkQueue這個對象中了。
  super(corePoolSize, Integer.MAX_VALUE, 0, NANOSECONDS,
    new DelayedWorkQueue());
 }

2.DelayedWorkQueue

DelayedWorkQueue是在ScheduledThreadPoolExecutor的一個內部類,實現(xiàn)了BlockingQueue接口
里面存放任務隊列的數(shù)組如下:

private RunnableScheduledFuture<?>[] queue =
   new RunnableScheduledFuture<?>[INITIAL_CAPACITY];

我們分析過ThreadPoolExecutor,它從任務隊列中獲取任務的方式為poll和take兩種,所以看一下poll和take兩個方法的源碼,回顧一下,ThreadPoolExecutor它會調用poll或take方法,先poll,再take,只要其中一個接口有返回就行

public RunnableScheduledFuture<?> poll() {
   final ReentrantLock lock = this.lock;
   lock.lock();
   try {
    RunnableScheduledFuture<?> first = queue[0];
    // 這里有個getDelay,這是關鍵點,獲取執(zhí)行延時時間
    // 但是如果我們有延時設置的話,這就返回空了,然后就會調用take方法
    if (first == null || first.getDelay(NANOSECONDS) > 0)
     return null;
    else
     return finishPoll(first);
   } finally {
    lock.unlock();
   }
  }

public RunnableScheduledFuture<?> take() throws InterruptedException {
   final ReentrantLock lock = this.lock;
   lock.lockInterruptibly();
   try {
    for (;;) {
     RunnableScheduledFuture<?> first = queue[0];
     if (first == null)
      available.await();
     else {
     // 獲取延時時間
      long delay = first.getDelay(NANOSECONDS);
      if (delay <= 0)
       return finishPoll(first);
      first = null; // don't retain ref while waiting
      if (leader != null)
       available.await();
      else {
       Thread thisThread = Thread.currentThread();
       leader = thisThread;
       try {
       // 使用鎖,執(zhí)行延時等待。
       // 使用鎖,執(zhí)行延時等待。
       // 使用鎖,執(zhí)行延時等待。
        available.awaitNanos(delay);
       } finally {
        if (leader == thisThread)
         leader = null;
       }
      }
     }
    }
   } finally {
    if (leader == null && queue[0] != null)
     available.signal();
    lock.unlock();
   }
  }

3.RunnableScheduledFuture

在ScheduledThreadPoolExecutor內部有一個ScheduledFutureTask類實現(xiàn)了RunnableScheduledFuture,ScheduledFutureTask這個類采用了裝飾者設計模式,在執(zhí)行Runnable的方法基礎上還執(zhí)行了一些額外的功能。
我們需要特別注意幾個參數(shù)period、time。

(1)time

首先看一下time的作用,可以發(fā)現(xiàn)time是用于獲取執(zhí)行延時時間的,也就是delay是根據(jù)time生成的

public long getDelay(TimeUnit unit) {
   return unit.convert(time - now(), NANOSECONDS);
  }

(2)period

這個參數(shù)不是說設置執(zhí)行幾個周期,而是用于判斷是否需要按周期執(zhí)行,以及執(zhí)行周期,也就是本次執(zhí)行與下次執(zhí)行間隔的時間

// 判斷是否需要按周期執(zhí)行,如果周期設置成0,不是無間隔執(zhí)行,而是只執(zhí)行一次,這個需要特別注意
 public boolean isPeriodic() {
   return period != 0;
  }
 private void setNextRunTime() {
   long p = period;
   if (p > 0)
   // 這里將周期加給time,這樣獲取的延遲時間就是周期時間了。
    time += p;
   else
    time = triggerTime(-p);
  }

(3)執(zhí)行

 public void run() {
   // 先判斷是否為周期性的任務
   boolean periodic = isPeriodic();
   if (!canRunInCurrentRunState(periodic))
    cancel(false);
   else if (!periodic)
   // 如果不是周期性的,就執(zhí)行調用父類的run方法,也就是構造函數(shù)中傳入的Runnable對象的run方法。
    ScheduledFutureTask.super.run();
    // 在if的括號中先執(zhí)行了任務
   else if (ScheduledFutureTask.super.runAndReset()) {
   // 如果是周期性的,就需要設置下次執(zhí)行的時間,然后利用reExecutePeriodic方法,將任務再次丟入任務隊列中。
   // 這里尤其需要注意的是if中的邏輯執(zhí)行失敗,如果沒有捕捉異常,那么后面的邏輯就不會再執(zhí)行了,也就是說中間有一次執(zhí)行失敗,后面這個周期性的任務就失效了。
    setNextRunTime();
    reExecutePeriodic(outerTask);
   }
  }

總結

ScheduledThreadPoolExecutor通過time參數(shù),設置當前任務執(zhí)行的等待時間,再通過period設置任務下次執(zhí)行需要等待的時間。這兩個參數(shù)都不是設置在線程池中的,而是攜帶在任務中的,這就可以把線程池和任務進行完全解耦。
注意點:
(1)任務的執(zhí)行等待時間是在隊列的take方法中的。
(2)period參數(shù)設置成0,任務將只會執(zhí)行一次,而不會執(zhí)行多次
(3)如果要自己實現(xiàn)周期性Task,周期性任務在執(zhí)行過程中,一定要注意捕捉異常,否則某一次執(zhí)行失敗,將導致后續(xù)的任務周期失效,任務將不再繼續(xù)執(zhí)行。

到此這篇關于java 定時器線程池(ScheduledThreadPoolExecutor)的實現(xiàn)的文章就介紹到這了,更多相關java 定時器線程池內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • maven無法依賴spring-cloud-stater-zipkin的解決方案

    maven無法依賴spring-cloud-stater-zipkin的解決方案

    這篇文章主要介紹了maven無法依賴spring-cloud-stater-zipkin如何解決,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2023-05-05
  • Java中綴表達式轉后綴表達式流程詳解

    Java中綴表達式轉后綴表達式流程詳解

    中綴表達式是一個通用的算術或邏輯公式表示方法。,中綴表達式不容易被計算機解析,但仍被許多程序語言使用,因為它符合人們的普遍用法。本文介紹了實現(xiàn)中綴表達式的方法,需要的可以參考一下
    2022-09-09
  • 淺談java socket的正確關閉姿勢

    淺談java socket的正確關閉姿勢

    這篇文章主要介紹了java socket的正確關閉姿勢,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-06-06
  • Java面試題沖刺第十天--MyBatis2

    Java面試題沖刺第十天--MyBatis2

    這篇文章主要為大家分享了最有價值的三道MyBatis框架面試題,涵蓋內容全面,包括數(shù)據(jù)結構和算法相關的題目、經典面試編程題等,感興趣的小伙伴們可以參考一下
    2021-07-07
  • 詳解java整合solr5.0之solrj的使用

    詳解java整合solr5.0之solrj的使用

    本篇文章主要介紹了詳解java整合solr5.0之solrj的使用 ,具有一定的參考價值,有興趣的可以了解下
    2017-06-06
  • Java 實戰(zhàn)項目錘煉之在線購書商城系統(tǒng)的實現(xiàn)流程

    Java 實戰(zhàn)項目錘煉之在線購書商城系統(tǒng)的實現(xiàn)流程

    讀萬卷書不如行萬里路,只學書上的理論是遠遠不夠的,只有在實戰(zhàn)中才能獲得能力的提升,本篇文章手把手帶你用java+jsp+mysql+servlet+ajax實現(xiàn)一個在線購書商城系統(tǒng),大家可以在過程中查缺補漏,提升水平
    2021-11-11
  • Spring整合SpringMVC與Mybatis(SSM)實現(xiàn)完整登錄功能流程詳解

    Spring整合SpringMVC與Mybatis(SSM)實現(xiàn)完整登錄功能流程詳解

    開學學校開始講servlet 后期要求做一個登錄功能,這個使用SSM先只做個簡單的只帶登錄功能的,且項目使用了MyBatis-Plus來簡化開發(fā)流程。看情況決定要不要升級功能或者換個寫法
    2022-09-09
  • EDI中JAVA通過FTP工具實現(xiàn)文件上傳下載實例

    EDI中JAVA通過FTP工具實現(xiàn)文件上傳下載實例

    這篇文章主要介紹了EDI中JAVA通過FTP工具實現(xiàn)文件上傳下載實例,具有一定的參考價值,有需要的可以了解一下。
    2016-11-11
  • Spring5學習之基礎知識總結

    Spring5學習之基礎知識總結

    這篇文章主要介紹了Spring5學習之基礎知識總結,文中有非常詳細的代碼示例,對正在學習java的小伙伴們有非常好的幫助,需要的朋友可以參考下
    2021-05-05
  • Nacos配置文件使用經驗及CAP原則詳解

    Nacos配置文件使用經驗及CAP原則詳解

    這篇文章主要為大家介紹了Nacos配置文件使用經驗及CAP規(guī)則詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2024-02-02

最新評論