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

Java并發(fā)Timer源碼分析

 更新時間:2018年07月12日 08:13:52   作者:狂小白  
這篇文章講述了java并發(fā)編程的相關知識點,并通過Timer源碼分析更深入的講解了java并發(fā)編程。

timer在JDK里面,是很早的一個API了。具有延時的,并具有周期性的任務,在newScheduledThreadPool出來之前我們一般會用Timer和TimerTask來做,但是Timer存在一些缺陷,為什么這么說呢?

Timer只創(chuàng)建唯一的線程來執(zhí)行所有Timer任務。如果一個timer任務的執(zhí)行很耗時,會導致其他TimerTask的時效準確性出問題。例如一個TimerTask每10秒執(zhí)行一次,而另外一個TimerTask每40ms執(zhí)行一次,重復出現的任務會在后來的任務完成后快速連續(xù)的被調用4次,要么完全“丟失”4次調用。Timer的另外一個問題在于,如果TimerTask拋出未檢查的異常會終止timer線程。這種情況下,Timer也不會重新回復線程的執(zhí)行了;它錯誤的認為整個Timer都被取消了。此時已經被安排但尚未執(zhí)行的TimerTask永遠不會再執(zhí)行了,新的任務也不能被調度了。

這里做了一個小的 demo 來復現問題,代碼如下:

package com.hjc;

import java.util.Timer;
import java.util.TimerTask;

/**
 * Created by cong on 2018/7/12.
 */
public class TimerTest {
 //創(chuàng)建定時器對象
 static Timer timer = new Timer();

 public static void main(String[] args) {
 //添加任務1,延遲500ms執(zhí)行
 timer.schedule(new TimerTask() {

  @Override
  public void run() {
  System.out.println("---one Task---");
  try {
   Thread.sleep(1000);
  } catch (InterruptedException e) {
   e.printStackTrace();
  }
  throw new RuntimeException("error ");
  }
 }, 500);
 //添加任務2,延遲1000ms執(zhí)行
 timer.schedule(new TimerTask() {

  @Override
  public void run() {
  for (;;) {
   System.out.println("---two Task---");
   try {
   Thread.sleep(1000);
   } catch (InterruptedException e) {
   // TODO Auto-generated catch block
   e.printStackTrace();
   }
  }
  }
 }, 1000);

 }
}

如上代碼先添加了一個任務在 500ms 后執(zhí)行,然后添加了第二個任務在 1s 后執(zhí)行,我們期望的是當第一個任務輸出 ---one Task--- 后等待 1s 后第二個任務會輸出 ---two Task---,

但是執(zhí)行完畢代碼后輸出結果如下所示:

例子2,

public class Shedule {
 private static long start;

 public static void main(String[] args) {
  TimerTask task = new TimerTask() {
   public void run() {
    System.out.println(System.currentTimeMillis()-start);
    try{
     Thread.sleep(3000);
    }catch (InterruptedException e){
     e.printStackTrace();
    }
   }
  };

  TimerTask task1 = new TimerTask() {
   @Override
   public void run() {
    System.out.println(System.currentTimeMillis()-start);
   }
  };

  Timer timer = new Timer();
  start = System.currentTimeMillis();
  //啟動一個調度任務,1S鐘后執(zhí)行
  timer.schedule(task,1000);
  //啟動一個調度任務,3S鐘后執(zhí)行
  timer.schedule(task1,3000);


 }

}

上面程序我們預想是第一個任務執(zhí)行后,第二個任務3S后執(zhí)行的,即輸出一個1000,一個3000.

實際運行結果如下:

實際運行結果并不如我們所愿。世界結果,是過了4S后才輸出第二個任務,即4001約等于4秒。那部分時間時間到哪里去了呢?那個時間是被我們第一個任務的sleep所占用了。

現在我們在第一個任務中去掉Thread.sleep();這一行代碼,運行是否正確了呢?運行結果如下:

可以看到確實是第一個任務過了1S后執(zhí)行,第二個任務在第一個任務執(zhí)行完后過3S執(zhí)行了。

這就說明了Timer只創(chuàng)建唯一的線程來執(zhí)行所有Timer任務。如果一個timer任務的執(zhí)行很耗時,會導致其他TimerTask的時效準確性出問題。

Timer 實現原理分析

下面簡單介紹下 Timer 的原理,如下圖是 Timer 的原理模型介紹:

1.其中 TaskQueue 是一個平衡二叉樹堆實現的優(yōu)先級隊列,每個 Timer 對象內部有唯一一個 TaskQueue 隊列。用戶線程調用 timer 的 schedule 方法就是把 TimerTask 任務添加到 TaskQueue 隊列,在調用 schedule 的方法時候 long delay 參數用來說明該任務延遲多少時間執(zhí)行。

2.TimerThread 是具體執(zhí)行任務的線程,它從 TaskQueue 隊列里面獲取優(yōu)先級最小的任務進行執(zhí)行,需要注意的是只有執(zhí)行完了當前的任務才會從隊列里面獲取下一個任務而不管隊列里面是否有已經到了設置的 delay 時間,一個 Timer 只有一個 TimerThread 線程,所以可知 Timer 的內部實現是一個多生產者單消費者模型。

從實現模型可以知道要探究上面的問題只需看 TimerThread 的實現就可以了,TimerThread 的 run 方法主要邏輯源碼如下:

public void run() {
 try {
  mainLoop();
 } finally {
  // 有人殺死了這個線程,表現得好像Timer已取消
  synchronized(queue) {
   newTasksMayBeScheduled = false;
   queue.clear(); // 消除過時的引用
  }
 }
}
 private void mainLoop() {
  while (true) {
   try {
    TimerTask task;
    boolean taskFired;
    //從隊列里面獲取任務時候要加鎖
    synchronized(queue) {
     ......
    }
    if (taskFired) 
     task.run();//執(zhí)行任務
   } catch(InterruptedException e) {
   }
  }
 }

可知當任務執(zhí)行過程中拋出了除 InterruptedException 之外的異常后,唯一的消費線程就會因為拋出異常而終止,那么隊列里面的其他待執(zhí)行的任務就會被清除。所以 TimerTask 的 run 方法內最好使用 try-catch 結構 catch 主可能的異常,不要把異常拋出到 run 方法外。

其實要實現類似 Timer 的功能使用 ScheduledThreadPoolExecutor 的 schedule 是比較好的選擇。ScheduledThreadPoolExecutor 中的一個任務拋出了異常,其他任務不受影響的。

ScheduledThreadPoolExecutor 例子如下:

/**
 * Created by cong on 2018/7/12.
 */
public class ScheduledThreadPoolExecutorTest {
 static ScheduledThreadPoolExecutor scheduledThreadPoolExecutor = new ScheduledThreadPoolExecutor(1);

 public static void main(String[] args) {

  scheduledThreadPoolExecutor.schedule(new Runnable() {

   public void run() {
    System.out.println("---one Task---");
    try {
     Thread.sleep(1000);
    } catch (InterruptedException e) {
     e.printStackTrace();
    }
    throw new RuntimeException("error ");
   }

  }, 500, TimeUnit.MICROSECONDS);

  scheduledThreadPoolExecutor.schedule(new Runnable() {

   public void run() {
    for (int i =0;i<5;++i) {
     System.out.println("---two Task---");
     try {
      Thread.sleep(1000);
     } catch (InterruptedException e) {
      e.printStackTrace();
     }
    }

   }

  }, 1000, TimeUnit.MICROSECONDS);

  scheduledThreadPoolExecutor.shutdown();
 }
}

運行結果如下:

之所以 ScheduledThreadPoolExecutor 的其他任務不受拋出異常的任務的影響是因為 ScheduledThreadPoolExecutor 中的 ScheduledFutureTask 任務中 catch 掉了異常,但是在線程池任務的 run 方法內使用 catch 捕獲異常并打印日志是最佳實踐。

相關文章

  • JPA merge聯(lián)合唯一索引無效問題解決方案

    JPA merge聯(lián)合唯一索引無效問題解決方案

    這篇文章主要介紹了JPA merge聯(lián)合唯一索引無效問題解決方案,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2020-09-09
  • Springboot容器級后置處理器BeanDefinitionRegistryPostProcessor

    Springboot容器級后置處理器BeanDefinitionRegistryPostProcessor

    這篇文章主要介紹了Springboot容器級后置處理器BeanDefinitionRegistryPostProcessor,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習吧
    2023-01-01
  • Java中接口的多態(tài)詳解

    Java中接口的多態(tài)詳解

    大家好,本篇文章主要講的是Java中接口的多態(tài)詳解,感興趣的同學趕快來看一看吧,對你有幫助的話記得收藏一下
    2022-02-02
  • 簡單工廠模式_動力節(jié)點Java學院整理

    簡單工廠模式_動力節(jié)點Java學院整理

    這篇文章主要介紹了簡單工廠模式的相關資料,和大家一起學習靜態(tài)工廠方法模式,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-08-08
  • MyBatis-Plus updateById不更新null值的方法解決

    MyBatis-Plus updateById不更新null值的方法解決

    用Mybatis-Plus的updateById()來更新數據時,無法將字段設置為null值,更新后數據還是原來的值,本文就來詳細的介紹一下解決方法,具有一定的參考價值,感興趣的可以了解一下
    2023-08-08
  • Java8新特性之接口中的默認方法和靜態(tài)方法詳解

    Java8新特性之接口中的默認方法和靜態(tài)方法詳解

    今天帶大家學習的是Java8新特性的相關知識,文章圍繞著Java接口中的默認方法和靜態(tài)方法展開,文中有非常詳細的的代碼示例,需要的朋友可以參考下
    2021-06-06
  • Java Benchmark 基準測試的實例詳解

    Java Benchmark 基準測試的實例詳解

    這篇文章主要介紹了Java Benchmark 基準測試的實例詳解的相關資料,這里提供實例幫助大家學習理解這部分內容,需要的朋友可以參考下
    2017-08-08
  • 如何解決EasyExcel導出文件LocalDateTime報錯問題

    如何解決EasyExcel導出文件LocalDateTime報錯問題

    這篇文章主要介紹了如何解決EasyExcel導出文件LocalDateTime報錯問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-06-06
  • Java基于JDBC實現事務,銀行轉賬及貨物進出庫功能示例

    Java基于JDBC實現事務,銀行轉賬及貨物進出庫功能示例

    這篇文章主要介紹了Java基于JDBC實現事務,銀行轉賬及貨物進出庫功能,較為詳細的分析了事務操作的原理、實現方法及java基于jdbc連接數據庫實現銀行事務操作的相關技巧,需要的朋友可以參考下
    2017-12-12
  • 通過Java添加Word文本框過程詳解

    通過Java添加Word文本框過程詳解

    這篇文章主要介紹了通過Java添加Word文本框過程詳解,在Word中,文本框是指一種可移動、可調節(jié)大小的文字或圖形容器。我們可以向文本框中添加文字、圖片、表格等對象,下面,將通過Java編程來實現添加以上對象到Word文本框,需要的朋友可以參考下
    2019-07-07

最新評論