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

Java之定時器Timer和定時任務(wù)TimerTask應(yīng)用以及原理解讀

 更新時間:2024年12月23日 15:55:20   作者:zhangbeizhen18  
文章介紹了Java JDK自帶的定時器Timer和定時任務(wù)TimerTask的使用和原理,Timer和TimerTask成對出現(xiàn),Timer是定時器,TimerTask是定時任務(wù),TimerTask實現(xiàn)Runnable接口的run方法,Timer的屬性TimerThreadthread繼承Thread

記錄:272

場景:Java JDK自帶的定時器Timer和定時任務(wù)TimerTask應(yīng)用以及原理簡析。

在JDK工具包:java.util中可以找到源碼,即java.util.Timer和java.util.TimerTask。

TimerTask實現(xiàn)Runnable接口的run方法。

Timer的屬性TimerThread thread繼承Thread。因此,Timer天生就具備多線程屬性。

這個輕量級的定時器和定時任務(wù),在多線程的場景中使用極其便利和靈活。

版本

JDK 1.8
Spring Boot 2.6.3
Spring Framework 5.3.15

1.基礎(chǔ)

Timer和TimerTask成對出現(xiàn),Timer是定時器,TimerTask是定時任務(wù)。換句話說,定時任務(wù)TimerTask是給定時器Timer執(zhí)行的具體任務(wù)。

在JDK 1.8中TimerTask是抽象類(abstract),使用者繼承TimerTask,并實現(xiàn)抽象方法run()方法即可。TimerTask實現(xiàn)Runnable接口的run()。

1.1 TimerTask

TimerTask,即java.util.TimerTask,一個抽象類(abstract修飾的類),實現(xiàn)(implements)可被線程執(zhí)行的Runnable接口。

TimerTask需關(guān)注的內(nèi)容。

(1)屬性:final Object lock,多線程同步時使用。在Timer的TimerThread中同步鎖synchronized會使用。

(2)屬性:int state,標(biāo)識TimerTask的當(dāng)前狀態(tài)。包括VIRGIN,SCHEDULED,EXECUTED,CANCELLED。

(3)屬性:long nextExecutionTime,下一次執(zhí)行的時間。

(4)屬性:long period,重復(fù)任務(wù)的執(zhí)行頻率。

(5)方法:boolean cancel(),取消任務(wù),實際標(biāo)記任務(wù)狀態(tài)state為CANCELLED。

(6)方法:scheduledExecutionTime(),獲取調(diào)度時間。

(7)方法:protected TimerTask(),無參構(gòu)造方法。

(8)方法:abstract void run(),線程執(zhí)行具體函數(shù),使用者的具體業(yè)務(wù)任務(wù)就在這個方法中寫。

1.2 TaskQueue

TaskQueue,即java.util.TaskQueue。TaskQueue維護一個屬性TimerTask[] queue,這是Timer的任務(wù)線程操作的底層隊列。在Timer中有TaskQueue屬性,是給任務(wù)線程TimerThread使用的。

TaskQueue需關(guān)注的內(nèi)容。

(1)屬性:TimerTask[] queue,隊列中存放TimerTask。Timer的TimerThread通過遍歷queue,取出TimerTask,根據(jù)TimerTask屬性來確定當(dāng)前是否執(zhí)行。。

注意:使用者不需要直接操作TimerTask[] queue,它是交給Timer的TimerThread操作。

1.3 TimerThread

TimerThread,即java.util.TimerThread。繼承Thread類。因此,天生就具備多線程屬性。

TimerThread需關(guān)注的內(nèi)容。

(1)屬性:TaskQueue queue,TimerThread操作queue來找到當(dāng)前需要執(zhí)行的定時任務(wù)TimerTask。根據(jù)執(zhí)行策略執(zhí)行定時任務(wù)TimerTask中的run方法。

(2)方法:void run(),是TimerThread從Thread中繼承的方法。線程執(zhí)行的具體內(nèi)容,必須在這個方法中寫入或者在這個方法中被調(diào)用。TimerThread實現(xiàn)細節(jié)方法是void mainLoop()。在run方法中會調(diào)用mainLoop(),這樣定時任務(wù)就被任務(wù)線程調(diào)用到。

(3)方法:void mainLoop(),TimerThread中邏輯細節(jié)落地方法。本方法內(nèi)容在while (true)中,只要隊列有任務(wù)就會無限循環(huán)。方法核心內(nèi)容就是遍歷TimerThread的屬性:TaskQueue queue,找到符合條件的需執(zhí)行的TimerTask,并執(zhí)行TimerTask的run方法。

(4)方法: void start(),此方法是TimerThread從Thread繼承來的,作用就是啟動線程。TimerThread的start()方在Timer創(chuàng)建時就會調(diào)用。Timer創(chuàng)建后,TimerThread就啟動了。

1.4 Timer

Timer,即java.util.Timer。使用者操定時器就是操作這個類的任務(wù)調(diào)度方法。

Timer的需關(guān)注的內(nèi)容。

(1)屬性:final TaskQueue queue,Timer存放所有的定時任務(wù)的隊列,定時器Timer的核心就是遍歷這個隊列,找到當(dāng)前時間下,符合條件的需執(zhí)行的定時任務(wù)TimerTask。

(2)屬性:final TimerThread thread,Timer執(zhí)行定時任務(wù)的后臺線程。定時任務(wù)具體執(zhí)行邏輯在TimerThread的run方法中。TimerThread 線程在Timer的構(gòu)造函數(shù)中會調(diào)用TimerThread的start()啟動線程。

(3)構(gòu)造函數(shù)

  • 1>Timer(),無參構(gòu)造函數(shù)。
  • 2>Timer(boolean isDaemon),有參構(gòu)造函數(shù)。設(shè)置定時器線程是否為守護線程。就是設(shè)置屬性:TimerThread thread,調(diào)用Thread的setDaemon()方法。
  • 3>Timer(String name),有參構(gòu)造函數(shù)。設(shè)置線程名稱。
  • 4>Timer(String name, boolean isDaemon)。有參構(gòu)造函數(shù)。設(shè)置線程名稱和設(shè)置是否為守護線程。

(4)任務(wù)調(diào)度函數(shù)

  • 1>任務(wù)調(diào)度函數(shù)

操作Timer定時器,實際就是操作如下任務(wù)調(diào)度函數(shù)。

public void schedule(TimerTask task, long delay);
public void schedule(TimerTask task, Date time);
public void schedule(TimerTask task, long delay, long period);
public void schedule(TimerTask task, Date firstTime, long period); 
public void scheduleAtFixedRate(TimerTask task, long delay, long period);
public void scheduleAtFixedRate(TimerTask task, Date firstTime,long period);
  • 2>任務(wù)調(diào)度函數(shù)(sched)

任務(wù)調(diào)度函數(shù)都會調(diào)用一個私有方法sched。

private void sched(TimerTask task, long time, long period);
  • TimerTask task:繼承抽象類TimerTask,重寫abstract void run()方法,把具體業(yè)務(wù)任務(wù)寫在這個方法里。
  • long time:調(diào)度任務(wù)的起始時間值。調(diào)度是從這個時間為計算起點。
  • long period:任務(wù)調(diào)度頻率。

把定時任務(wù)TimerTask添加到隊列,就是在這個方法中實現(xiàn)。

(5)方法:void cancel(),取消隊列里所有任務(wù)。

(6)方法:void purge(),清除隊列里所有任務(wù)。

2.Timer原理解析

Timer原理解析,來自JDK中java.util.Timer源碼邏輯。如有疑惑,可直接翻閱源碼,便一目了然。

2.1 添加定時任務(wù)

把TimerTask添加到Timer內(nèi)部的TaskQueue中。

2.2.1 自定義實現(xiàn)TimerTask的 run()

自定義實現(xiàn)抽象類TimerTask的 run()方法,即任務(wù)需要執(zhí)行的業(yè)務(wù)邏輯寫在run方法中。

2.2.2 創(chuàng)建定時器Timer

創(chuàng)建定時器Timer,會做三件核心事情。

(1)創(chuàng)建任務(wù)隊列:TaskQueue queue。

(2)創(chuàng)建定時器線程:TimerThread thread。

(3)啟動定時器線程:執(zhí)行TimerThread的start()方法,即后臺循環(huán)掃描的定時器線程啟動。

2.2.3 調(diào)用Timer任務(wù)調(diào)度函數(shù)

添加定時任務(wù)就在調(diào)用Timer任務(wù)調(diào)度方法中完成。

(1)設(shè)置定時器任務(wù)調(diào)度參數(shù)。

(2)把自定義任務(wù)傳遞給任務(wù)調(diào)度函數(shù)。

2.2.4 Timer任務(wù)調(diào)度函數(shù)核心邏輯

Timer暴露給使用者的所有任務(wù)調(diào)度函數(shù),最終都會落地在一個私有方法,即

void sched(TimerTask task, long time, long period);

核心邏輯如下:

(1)同步鎖:synchronized(queue),鎖住隊列。隊列所有操作在此鎖內(nèi)完成。

(2)同步鎖:synchronized(task.lock),鎖住任務(wù)。

此鎖內(nèi)主要操作:設(shè)置TimerTask任務(wù)的執(zhí)行時間,執(zhí)行頻率,任務(wù)狀態(tài)。操作完成立即解鎖。

(3)隊列TaskQueue queue,把設(shè)置好的TimerTask添加到TaskQueue 中。

(4)解除同步鎖:synchronized(queue)。

2.2 執(zhí)行定時任務(wù)

執(zhí)行定時任務(wù),在Timer的定時器線程TimerThread thread中,即線程的run()方法中,最終落地方式是 mainLoop()。注意mainLoop()是在run()方法中調(diào)用,是邏輯集中寫在mainLoop()中,增加代碼易讀性。

2.2.1 while (true)入口

mainLoop()入口是while (true),即使循環(huán)掃描。

2.2.2 同步鎖:synchronized(queue)

同步鎖:synchronized(queue),鎖住隊列,一個Timer共用一個隊列,因此使用synchronized有效。

2.2.3 判斷隊列是否為空

使用 while (queue.isEmpty() && newTasksMayBeScheduled)判斷隊列是否為空,如果為空,則 queue.wait()等待,注意wait()是java.lang.Object的方法,因此,此時while在卡主狀態(tài),直到queue.notify()被調(diào)用,才會繼續(xù)。

2.2.4 確認隊列為空跳出循環(huán)

使用if (queue.isEmpty()),判斷隊列確定為空了,那么就break跳出循環(huán),其實任務(wù)線程就優(yōu)雅結(jié)束了。

2.2.5 取出一個任務(wù)TimerTask

取出一個任務(wù):task = queue.getMin();

2.2.6 同步鎖:synchronized(task.lock),操作任務(wù)

使用同步鎖:synchronized(task.lock),鎖住任務(wù)。判斷任務(wù)是否可執(zhí)行。

(1)判斷任務(wù)狀態(tài)。

(2)取當(dāng)前系統(tǒng)時間:System.currentTimeMillis()。

(3)取出的任務(wù)執(zhí)行時間: task.nextExecutionTime。

(4)判斷當(dāng)前系統(tǒng)時間和任務(wù)執(zhí)行時間,來確定任務(wù)是否需要執(zhí)行。

(5)使用任務(wù)狀態(tài)標(biāo)識taskFired=true任務(wù)需啟動;否則,不啟動。。

(6)任務(wù)操作完成,解除同步鎖:synchronized(task.lock)。

2.2.7 解除同步鎖:synchronized(queue)

2.2.8 執(zhí)行任務(wù):task.run()。

自定義的定時任務(wù)TimerTask,在此處就被執(zhí)行。

3.案例

本例每隔60秒,獲取一次UUID。

(1)實現(xiàn)TimerTask的run方法,把具體執(zhí)行的業(yè)務(wù)任務(wù)寫到run方法中。

(2)創(chuàng)建Timer對象,配置定時任務(wù)和傳入TimerTask對象。

Timer提供多種任務(wù)調(diào)度策略,本例使用固定頻率任務(wù)調(diào)度。

TimeMonitor,實現(xiàn)了InitializingBean接口,在afterPropertiesSet中初始化定時器。也就是TimeMonitor被容器初始化完成后,就會觸發(fā)Timer定時器初始化,Timer定時器內(nèi)部的任務(wù)線程會啟動和任務(wù)隊列會把使用者的定時任務(wù)添加到隊列。Timer定時器就會生效運行。

@Slf4j
@Service
public class TimeMonitor implements InitializingBean {
  @Override
  public void afterPropertiesSet() throws Exception {
      startTimerMonitor();
  }
  private void startTimerMonitor() {
    TimerTask timerTask = new TimerTask() {
      @Override
      public void run() {
          log.info("定時任務(wù)執(zhí)行開始.");
          String uuid = UUID.randomUUID().toString()
              .replace("-", "").toUpperCase();
          log.info("執(zhí)行業(yè)務(wù),獲取序列號,UUID = " + uuid);
          log.info("定時任務(wù)執(zhí)行完成.");
      }
    };
    // 定時器
    Timer timer = new Timer();
    // 延時時間
    long delayTime = 6000L;
    // 執(zhí)行頻率
    long period = 1000 * 60;
    timer.scheduleAtFixedRate(timerTask, delayTime, period);
  }
}

4.測試

根據(jù)輸出日志查看定時調(diào)度情況。

總結(jié)

以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • ZooKeeper入門教程三分布式鎖實現(xiàn)及完整運行源碼

    ZooKeeper入門教程三分布式鎖實現(xiàn)及完整運行源碼

    本文是ZooKeeper入門系列教程,分布式鎖有多種實現(xiàn)方式,比如通過數(shù)據(jù)庫、redis都可實現(xiàn)。作為分布式協(xié)同工具ZooKeeper,當(dāng)然也有著標(biāo)準(zhǔn)的實現(xiàn)方式。本文介紹在zookeeper中如何實現(xiàn)排他鎖
    2022-01-01
  • 解決IDEA?2022?Translation?翻譯文檔失敗:?未知錯誤的問題

    解決IDEA?2022?Translation?翻譯文檔失敗:?未知錯誤的問題

    這篇文章主要介紹了IDEA?2022?Translation?翻譯文檔失敗:?未知錯誤,本文較詳細的給大家介紹了IDEA?2022?Translation未知錯誤翻譯文檔失敗的解決方法,需要的朋友可以參考下
    2022-04-04
  • Java實現(xiàn)自定義自旋鎖代碼實例

    Java實現(xiàn)自定義自旋鎖代碼實例

    這篇文章主要介紹了Java實現(xiàn)自定義自旋鎖代碼實例,Java自旋鎖是一種線程同步機制,它允許線程在獲取鎖時不立即阻塞,而是通過循環(huán)不斷嘗試獲取鎖,直到成功獲取為止,自旋鎖適用于鎖競爭激烈但持有鎖的時間很短的情況,需要的朋友可以參考下
    2023-10-10
  • 深入聊聊Java內(nèi)存泄露問題

    深入聊聊Java內(nèi)存泄露問題

    所謂內(nèi)存泄露就是指一個不再被程序使用的對象或變量一直被占據(jù)在內(nèi)存中,下面這篇文章主要給大家介紹了關(guān)于Java內(nèi)存泄露問題的相關(guān)資料,文中通過示例代碼介紹的非常詳細,需要的朋友可以參考下
    2022-04-04
  • Spring?Boot?Actuator管理日志的實現(xiàn)

    Spring?Boot?Actuator管理日志的實現(xiàn)

    本文主要介紹了Spring?Boot?Actuator管理日志的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-07-07
  • 詳解MyBatis特性之動態(tài)SQL

    詳解MyBatis特性之動態(tài)SQL

    動態(tài) SQL 是 MyBatis 的強大特性之一,這篇文章我們將結(jié)合動態(tài)SQL完成更加復(fù)雜的 SQL 操作,文章通過代碼示例給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下
    2024-01-01
  • Maven2環(huán)境安裝與準(zhǔn)備工作詳解

    Maven2環(huán)境安裝與準(zhǔn)備工作詳解

    這篇文章主要為大家詳細介紹了Maven2環(huán)境安裝與準(zhǔn)備工作,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-06-06
  • java實現(xiàn)學(xué)生信息錄入界面

    java實現(xiàn)學(xué)生信息錄入界面

    這篇文章主要為大家詳細介紹了java實現(xiàn)學(xué)生信息錄入界面,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-04-04
  • shiro之INI配置詳解

    shiro之INI配置詳解

    這篇文章主要為大家詳細介紹了shiro之INI配置的相關(guān)資料,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-09-09
  • Java 詳解如何獲取網(wǎng)絡(luò)接口信息

    Java 詳解如何獲取網(wǎng)絡(luò)接口信息

    讀萬卷書不如行萬里路,只學(xué)書上的理論是遠遠不夠的,只有在實踐中才能獲得能力的提升,本篇文章手把手帶你用Java獲取網(wǎng)絡(luò)接口的信息,大家可以在過程中查缺補漏,提升水平
    2021-11-11

最新評論