JavaEE線程安全定時(shí)器模式任務(wù)
前言
像是一個(gè)鬧鐘定時(shí),在一定時(shí)間之后被喚醒并執(zhí)行某個(gè)之前設(shè)定好的任務(wù),join(指定超時(shí)時(shí)間),sleep(指定休眠時(shí)間)都是基于系統(tǒng)內(nèi)部的定時(shí)器來實(shí)現(xiàn)的。
java.util.Timer核心方法就一個(gè),schedule參數(shù)有兩個(gè):任務(wù)是啥(一段代碼),多長(zhǎng)時(shí)間之后執(zhí)行
public class 定時(shí)器 { ? ? public static void main(String[] args) { ? ? ? ? Timer timer = new Timer(); ? ? ? ? timer.schedule(new TimerTask() { ? ? ? ? ? ? @Override ? ? ? ? ? ? public void run() { ? ? ? ? ? ? ? ? System.out.println("hello timer"); ? ? ? ? ? ? } ? ? ? ? }, 3000); ? ? } }
Timer內(nèi)部組成
1.描述任務(wù)
創(chuàng)建一個(gè)專門的類來表示一個(gè)定時(shí)器的任務(wù)TimerTask,這個(gè)MyTask類的比較規(guī)則不是默認(rèn)存在的,需要我們手動(dòng)指定按照時(shí)間大小來比較的。
//創(chuàng)建一個(gè)類表示一個(gè)任務(wù) class MyTask{ ? ? //任務(wù)描述 ? ? private Runnable runnable; ? ? //任務(wù)具體啥時(shí)候干 ? ? private long time; ? ? //after是一個(gè)時(shí)間間隔,不是覺得時(shí)間戳的值 ? ? public MyTask(Runnable runnable, long after){ ? ? ? ? this.runnable = runnable; ? ? ? ? this.time = System.currentTimeMillis() + after; ? ? } ? ? public void run(){ ? ? ? ? runnable.run(); ? ? } }
2.組織任務(wù)
通過一定的數(shù)據(jù)結(jié)構(gòu)把一些任務(wù)給放到一起,標(biāo)準(zhǔn)庫(kù)有一個(gè)專門的數(shù)據(jù)結(jié)構(gòu)PriorityQueue,這里用到的數(shù)據(jù)結(jié)構(gòu)是PriorityBlockingQueue,及帶有優(yōu)先級(jí)又帶有阻塞隊(duì)列。此處的隊(duì)列要考慮到線程安全問題,可能在多個(gè)線程里進(jìn)行注冊(cè)任務(wù),同時(shí)還有一個(gè)專門的線程來取任務(wù)執(zhí)行,此時(shí)的隊(duì)列就需要注意線程安全問題。
其次為了避免盲等的現(xiàn)象,可以使用wait這樣的機(jī)制指定等待時(shí)間時(shí)間到了自然喚醒,計(jì)算出當(dāng)前時(shí)間和任務(wù)目標(biāo)的時(shí)間差即可。既然是指定一個(gè)等待時(shí)間為啥不用sleep而是用wait,sleep不能被中途喚醒,wait能夠被中途喚醒。在等待過程中可能有新的任務(wù)插入,新的任務(wù)是可能出現(xiàn)在之前所有任務(wù)的最前面,在schedule操作中就需要加上一個(gè)notify操作。
3.執(zhí)行時(shí)間到了的任務(wù)
需要執(zhí)行時(shí)間最靠前的任務(wù),就需要一個(gè)線程不停地去檢查當(dāng)前優(yōu)先隊(duì)列的對(duì)手元素,看看當(dāng)前最靠前的任務(wù)是不是時(shí)間到了
class MyTask implements Comparable<MyTask>{ ? ? //任務(wù)描述 ? ? private Runnable runnable; ? ? //任務(wù)具體啥時(shí)候干 ? ? private long time; ? ? //delay是一個(gè)時(shí)間間隔,不是絕對(duì)時(shí)間戳的值 ? ? public MyTask(Runnable runnable, long delay){ ? ? ? ? this.runnable = runnable; ? ? ? ? this.time = System.currentTimeMillis() + delay; ? ? } ? ? public void run(){ ? ? ? ? runnable.run(); ? ? } ? ? public long getTime(){ ? ? ? ? return time; ? ? } ? ? @Override ? ? public int compareTo(MyTask o) { ? ? ? ? //小的在前 ? ? ? ? return (int)(this.time - o.time); ? ? } } class MyTimer{ ? ? private Object locker = new Object(); ? ? //定時(shí)器內(nèi)不要能夠存放多個(gè)任務(wù) ? ? private PriorityBlockingQueue<MyTask> queue = new PriorityBlockingQueue<>(); ? ? public void schedule(Runnable runnable, long delay){ ? ? ? ? MyTask task = new MyTask(runnable, delay); ? ? ? ? queue.put(task); ? ? ? ? //每次任務(wù)插入成功之后都喚醒一下掃描線程,讓線程重新檢查隊(duì)首的任務(wù)是否時(shí)間到要執(zhí)行 ? ? ? ? synchronized (locker){ ? ? ? ? ? ? locker.notify(); ? ? ? ? } ? ? } ? ? public MyTimer(){ ? ? ? ? Thread t = new Thread(() ->{ ? ? ? ? ? ? while(true){ ? ? ? ? ? ? ? ? try{ ? ? ? ? ? ? ? ? ? ? //先取出隊(duì)首元素 ? ? ? ? ? ? ? ? ? ? MyTask task = queue.take(); ? ? ? ? ? ? ? ? ? ? //在比較一下看看當(dāng)前這個(gè)任務(wù)時(shí)間到了沒 ? ? ? ? ? ? ? ? ? ? long curTime = System.currentTimeMillis(); ? ? ? ? ? ? ? ? ? ? if(curTime < task.getTime()){ ? ? ? ? ? ? ? ? ? ? ? ? //時(shí)間沒到把任務(wù)賽回到隊(duì)列中 ? ? ? ? ? ? ? ? ? ? ? ? queue.put(task); ? ? ? ? ? ? ? ? ? ? ? ? //制定一個(gè)等待時(shí)間 ? ? ? ? ? ? ? ? ? ? ? ? synchronized (locker){ ? ? ? ? ? ? ? ? ? ? ? ? ? ? locker.wait(task.getTime() - curTime); ? ? ? ? ? ? ? ? ? ? ? ? } ? ? ? ? ? ? ? ? ? ? }else{ ? ? ? ? ? ? ? ? ? ? ? ? //時(shí)間到了執(zhí)行任務(wù) ? ? ? ? ? ? ? ? ? ? ? ? task.run(); ? ? ? ? ? ? ? ? ? ? } ? ? ? ? ? ? ? ? } catch (InterruptedException e) { ? ? ? ? ? ? ? ? ? ? e.printStackTrace(); ? ? ? ? ? ? ? ? } ? ? ? ? ? ? } ? ? ? ? }); ? ? ? ? t.start(); ? ? } } public class 定時(shí)器 { ? ? public static void main(String[] args) { ? ? ? ? MyTimer myTimer = new MyTimer(); ? ? ? ? myTimer.schedule(new Runnable() { ? ? ? ? ? ? @Override ? ? ? ? ? ? public void run() { ? ? ? ? ? ? ? ? System.out.println("hello timer"); ? ? ? ? ? ? } ? ? ? ? }, 3000); ? ? ? ? myTimer.schedule(new Runnable() { ? ? ? ? ? ? @Override ? ? ? ? ? ? public void run() { ? ? ? ? ? ? ? ? System.out.println("hello "); ? ? ? ? ? ? } ? ? ? ? }, 2000); ? ? } }
到此這篇關(guān)于JavaEE線程安全定時(shí)器模式任務(wù)的文章就介紹到這了,更多相關(guān)JavaEE 定時(shí)器 內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Spring MVC深入學(xué)習(xí)之啟動(dòng)初始化過程
最近因?yàn)楣ぷ鞯脑蛟趯W(xué)習(xí)Spring MVC,為了更深入的學(xué)習(xí)Spring MVC,下面這篇文章主要給大家介紹了關(guān)于Spring MVC深入學(xué)習(xí)之啟動(dòng)初始化過程的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起看看吧。2017-07-07理解 MyBatis 是如何在 Spring 容器中初始化的
這篇文章主要介紹了理解 MyBatis 是如何在 Spring 容器中初始化的,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-11-11Spring boot按日切分spring boot的nohup.out日志文件的方法
過大的日志文件維護(hù)起來存在諸多問題,所以最好是能夠按日或按大小切分日志文件,下面小編給大家?guī)砹薙pring boot按日切分spring boot的nohup.out日志文件的方法,一起看看吧2018-08-08JAVA使用commos-fileupload實(shí)現(xiàn)文件上傳與下載實(shí)例解析
這篇文章主要介紹了JAVA使用commos-fileupload實(shí)現(xiàn)文件上傳與下載的相關(guān)資料,需要的朋友可以參考下2016-02-02一文帶你看懂Java8中的lambda表達(dá)式和方法引用
Lambda 表達(dá)式是 Java 8 引入的一項(xiàng)重要特性,它提供了一種簡(jiǎn)潔、清晰且靈活的語(yǔ)法來表示可傳遞的匿名函數(shù),下面就跟隨小編一起學(xué)習(xí)一下Java8中的lambda表達(dá)式和方法引用的相關(guān)知識(shí)吧2023-12-12怎樣通過反射獲取非靜態(tài)內(nèi)部類實(shí)例
這篇文章主要介紹了怎樣通過反射獲取非靜態(tài)內(nèi)部類實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-03-03Spring Validation和Hibernate Validator結(jié)合國(guó)際化代碼實(shí)例
這篇文章主要介紹了Spring Validation和Hibernate Validator結(jié)合國(guó)際化代碼實(shí)例,我們需要對(duì)請(qǐng)求參數(shù)進(jìn)行非空、長(zhǎng)度、正確性進(jìn)行校驗(yàn), 本文主要講解Spring Validation 和 Hibernate Validator, 同時(shí)整合i18n(國(guó)際化)實(shí)現(xiàn)參數(shù)校驗(yàn)自動(dòng),需要的朋友可以參考下2023-10-10詳解java代碼中init method和destroy method的三種使用方式
這篇文章主要介紹了詳解java代碼中init method和destroy method的三種使用方式,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-03-03