Java并發(fā)教程之Callable和Future接口詳解
剛把Thread 的知識(shí)理了一遍。
Runnable是一個(gè)接口,而Thread是Runnable的一個(gè)實(shí)現(xiàn)類。
所以也就有了之前創(chuàng)建線程的兩種方法
- 繼承Thread
- 實(shí)現(xiàn)Runnable
我們看一下新建線程的方法:
都是得傳入一個(gè)Runnable對(duì)象(這句話很關(guān)鍵)
所以傳入一個(gè)Runnble和Thread對(duì)象都行。
現(xiàn)在引入創(chuàng)建線程的第三種方法:Callable
為了實(shí)現(xiàn) Runnable,需要實(shí)現(xiàn)不返回任何內(nèi)容的 run()方法,而對(duì)于
Callable,需要實(shí)現(xiàn)在完成時(shí)返回結(jié)果的 call()方法。
• call()方法可以引發(fā)異常,而 run()則不能。
Callable最大的特點(diǎn)就是 它能返回?cái)?shù)值,并拋出異常,而且它不是run()方法,而是call()方法。
但是Callable最大的問題就是不能丟人Thread類中直接創(chuàng)建,因?yàn)門hread只認(rèn)識(shí)Runnable,所以我們必須找到Callable和Runnable之間的橋梁。
也就是找一個(gè)類,即和Runnable友關(guān)系,又和Callable有關(guān)系。
這個(gè)類就是FutureTask了。
- Runnable接口有實(shí)現(xiàn)類FutureTask
- FutureTask構(gòu)造可以傳遞Callable
所以創(chuàng)建也給Callable可以這樣
新類 MyThread2 實(shí)現(xiàn) callable 接口 class MyThread2 implements Callable<Integer>{ @Override public Integer call() throws Exception { return 200; }
1、新建一個(gè)callable對(duì)象
Callable callable = new MyThread2()
2、傳入callable對(duì)象,并新建一個(gè)futuretask對(duì)象,
FutureTask futureTask2 = new FutureTask(callable)
也可以之間通過lam表達(dá)式(函數(shù)式接口)實(shí)現(xiàn)
3、新建一個(gè)線程,傳入FutureTask對(duì)象
new Thread(futureTask2, “線程二”).start()
這就是基本步驟。
接下來可以來擴(kuò)展地講講 futuretask的作用
Future對(duì)象
Future 接口
當(dāng) call()方法完成時(shí),結(jié)果必須存儲(chǔ)在主線程已知的對(duì)象中,以便主線程可以知道該線程返回的結(jié)果。為此,可以使用 Future 對(duì)象。
將 Future 視為保存結(jié)果的對(duì)象–它可能暫時(shí)不保存結(jié)果,但將來會(huì)保存(一旦Callable 返回)。Future 基本上是主線程可以跟蹤進(jìn)度以及其他線程的結(jié)果的一種方式。要實(shí)現(xiàn)此接口,必須重寫 5 種方法,這里列出了重要的方法,如下:
• public boolean cancel(boolean mayInterrupt):用于停止任務(wù)。
如果尚未啟動(dòng),它將停止任務(wù)。如果已啟動(dòng),則僅在 mayInterrupt 為 true時(shí)才會(huì)中斷任務(wù)。
• public Object get()拋出 InterruptedException,ExecutionException:
用于獲取任務(wù)的結(jié)果。
如果任務(wù)完成,它將立即返回結(jié)果,否則將等待任務(wù)完成,然后返回結(jié)果。
• public boolean isDone():如果任務(wù)完成,則返回 true,否則返回 false可以看到 Callable 和 Future 做兩件事-Callable 與 Runnable 類似,因?yàn)樗庋b了要在另一個(gè)線程上運(yùn)行的任務(wù),而 Future 用于存儲(chǔ)從另一個(gè)線程獲得的結(jié)
果。實(shí)際上,future 也可以與 Runnable 一起使用。
要?jiǎng)?chuàng)建線程,需要 Runnable。為了獲得結(jié)果,需要 future。
FutureTask
Java 庫具有具體的 FutureTask 類型,該類型實(shí)現(xiàn) Runnable 和 Future,并方便地將兩種功能組合在一起。 可以通過為其構(gòu)造函數(shù)提供 Callable 來創(chuàng)建FutureTask。然后,將 FutureTask 對(duì)象提供給 Thread 的構(gòu)造函數(shù)以創(chuàng)建Thread 對(duì)象。因此,間接地使用 Callable 創(chuàng)建線程。
核心原理:(重點(diǎn))
在主線程中需要執(zhí)行比較耗時(shí)的操作時(shí),但又不想阻塞主線程時(shí),可以把這些作業(yè)交給 Future 對(duì)象在后臺(tái)完成
• 當(dāng)主線程將來需要時(shí),就可以通過 Future 對(duì)象獲得后臺(tái)作業(yè)的計(jì)算結(jié)果或者執(zhí)行狀態(tài)
• 一般 FutureTask 多用于耗時(shí)的計(jì)算,主線程可以在完成自己的任務(wù)后,再去獲取結(jié)果。
• 僅在計(jì)算完成時(shí)才能檢索結(jié)果;如果計(jì)算尚未完成,則阻塞 get 方法
• 一旦計(jì)算完成,就不能再重新開始或取消計(jì)算
• get 方法而獲取結(jié)果只有在計(jì)算完成時(shí)獲取,否則會(huì)一直阻塞直到任務(wù)轉(zhuǎn)入完成狀態(tài),然后會(huì)返回結(jié)果或者拋出異常
• get 只計(jì)算一次,因此 get 方法放到最后
舉個(gè)例子吧。
比如現(xiàn)在老師在上課,但是他渴了,他想買水,但是還得繼續(xù)上課。
這時(shí)就可以單獨(dú)開啟一條線程,讓班長(zhǎng)去幫他買水。(開啟單獨(dú)一條線程,不影響主線程的運(yùn)行)
get方法可以得到return的值
isDone()方法可以判斷是否完成了。
第一次會(huì)計(jì)算,第二次(之前計(jì)算過的),就直接返回結(jié)果了。
總結(jié)
到此這篇關(guān)于Java并發(fā)教程之Callable和Future接口的文章就介紹到這了,更多相關(guān)Java并發(fā)Callable和Future接口內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- Java使用多線程批次查詢大量數(shù)據(jù)(Callable返回?cái)?shù)據(jù))方式
- Java通過Callable實(shí)現(xiàn)多線程
- Java多線程中Callable和Future的解讀
- Java中的Callable實(shí)現(xiàn)多線程詳解
- Java使用Callable接口實(shí)現(xiàn)多線程的實(shí)例代碼
- Java多線程實(shí)現(xiàn)之Callable詳解
- Java中Runnable和Callable分別什么時(shí)候使用
- Java中Runnable與Callable接口的區(qū)別詳解
- 詳解Java中Callable和Future的區(qū)別
- Java使用Runnable和Callable實(shí)現(xiàn)多線程的區(qū)別詳解
- java面試常問的Runnable和Callable的區(qū)別
- Java中callable的實(shí)現(xiàn)原理
相關(guān)文章
SpringBoot使用maven指定依賴包的版本(解決示例)
我們?cè)谑褂肁依賴的時(shí)候,這個(gè)依賴有引入了第三方B依賴,這時(shí)候我想指定B依賴的版本號(hào),下面?zhèn)€大家分享解決示例,對(duì)SpringBoot maven依賴包相關(guān)配置方法感興趣的朋友一起看看吧2024-04-04SpringBoot集成xxl-job實(shí)現(xiàn)超牛的定時(shí)任務(wù)的步驟詳解
XXL-JOB是一個(gè)分布式任務(wù)調(diào)度平臺(tái),其核心設(shè)計(jì)目標(biāo)是開發(fā)迅速、學(xué)習(xí)簡(jiǎn)單、輕量級(jí)、易擴(kuò)展,現(xiàn)已開放源代碼并接入多家公司線上產(chǎn)品線,開箱即用,本文給大家介紹了SpringBoot集成xxl-job實(shí)現(xiàn)超牛的定時(shí)任務(wù),需要的朋友可以參考下2023-10-10SpringBoot項(xiàng)目@Async方法問題解決方案
這篇文章主要介紹了SpringBoot項(xiàng)目@Async方法問題解決方案,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-04-04SpringCloud Feign多參數(shù)傳遞及需要注意的問題
這篇文章主要介紹了SpringCloud Feign多參數(shù)傳遞及需要注意的問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-03-03Spring AOP如何自定義注解實(shí)現(xiàn)審計(jì)或日志記錄(完整代碼)
這篇文章主要介紹了Spring AOP如何自定義注解實(shí)現(xiàn)審計(jì)或日志記錄(完整代碼),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-12-12JAVA中常用的設(shè)計(jì)模式:?jiǎn)卫J?,工廠模式,觀察者模式
設(shè)計(jì)模式(Design pattern)代表了最佳的實(shí)踐,通常被有經(jīng)驗(yàn)的面向?qū)ο蟮能浖_發(fā)人員所采用。設(shè)計(jì)模式是軟件開發(fā)人員在軟件開發(fā)過程中面臨的一般問題的解決方案。這些解決方案是眾多軟件開發(fā)人員經(jīng)過相當(dāng)長(zhǎng)的一段時(shí)間的試驗(yàn)和錯(cuò)誤總結(jié)出來的。2020-04-04淺談Java當(dāng)作數(shù)組的幾個(gè)應(yīng)用場(chǎng)景
數(shù)組可以存放多個(gè)同一類型的數(shù)據(jù),可以存儲(chǔ)基本數(shù)據(jù)類型,引用數(shù)據(jù)類型(對(duì)象),下面這篇文章主要給大家介紹了關(guān)于Java當(dāng)作數(shù)組的幾個(gè)應(yīng)用場(chǎng)景,需要的朋友可以參考下2022-11-11Springboot中的異步任務(wù)執(zhí)行及監(jiān)控詳解
這篇文章主要介紹了Springboot中的異步任務(wù)執(zhí)行及監(jiān)控詳解,除了自己實(shí)現(xiàn)線程外,springboot本身就提供了通過注解的方式,進(jìn)行異步任務(wù)的執(zhí)行,下面主要記錄一下,在Springboot項(xiàng)目中實(shí)現(xiàn)異步任務(wù),以及對(duì)異步任務(wù)進(jìn)行封裝監(jiān)控,需要的朋友可以參考下2023-10-10logback的FileAppender文件追加模式和沖突檢測(cè)解讀
這篇文章主要為大家介紹了logback的FileAppender文件追加模式和沖突檢測(cè)解讀,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-10-10Spring AspectJ 實(shí)現(xiàn)AOP的方法你了解嗎
這篇文章主要為大家介紹了Spring AspectJ 實(shí)現(xiàn)AOP的方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助2022-01-01