中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Callable與Runnable有什么區別

發布時間:2021-06-16 14:26:22 來源:億速云 閱讀:302 作者:小新 欄目:大數據

這篇文章主要介紹了Callable與Runnable有什么區別,具有一定借鑒價值,感興趣的朋友可以參考下,希望大家閱讀完這篇文章之后大有收獲,下面讓小編帶著大家一起了解一下。

Callable 與 Runnable

java.lang.Runnable是一個接口,只有一個run()方法

public interface Runnable {
    public abstract void run();
}

run()方法的返回值是void,故在執行完任務后無法返回任何結果

Callable是java.util.concurrent包下的,也是一個接口,也只有一個call()方法,類似于java.lang.Runnable的run()方法,實現Callable接口的類和實現Runnable接口的類都是可以被其它線程執行的任務

public interface Callable<V> {
    V call() throws Exception;
}

可以看到call()方法是有返回值的,可以將執行的結果返回

Callable和Runnable的區別:
1、Callable中定義的是call()方法,Runnable中定義的是run()方法
2、Callable中的call()方法可以返回執行任務后的結果,Runnable中的run()方法無法獲得返回值
3、Callable中的call()方法定義了throws Exception拋出異常,拋出的異常可以在主線程Future.get()時被主線程捕獲;Runnable中的run()方法沒有定義拋出異常,運行任務時發生異常時也會上拋,因為即使不加默認也會上拋RuntimeException,但異常無法被主線程獲取
4、運行Callable任務可以拿到一個Future對象代表異步運算的結果

1、有了Runnable,為什么還需要Callable,它們的區別是什么?

Runnable和Callable都表示執行的任務,但不同的是Runnable.run()方法沒有返回值,Callable.call()有返回值
但其實線程在執行任務時還是執行的Runnable.run()方法,所以在使用ThreadPoolExecutor.submit()時會將Callable封裝為FutureTask,而FutureTask是Runnable和Future的實現類,所以在執行Callable的任務時,線程其實是執行FutureTask這個Runnable的run()方法,其中封裝了調用Callable.call()并返回結果的邏輯。

執行Runnable任務如果發生異常,主線程無法知曉;而執行Callable任務如果發生異常,在Future.get()時會拋出java.util.concurrent.ExecutionException,其中封裝了真實異常

2、Future.get()是如何獲取線程返回值的?

首先得益于Callable.call()方法定義了返回值,提交Callable任務后,Callable會被封裝成FutureTask,其既可以作為Runnable被執行,也可以作為Future獲取返回值,FutureTask.run()方法會調用Callable.call()中的任務代碼。在任務執行完成前,如果主線程使用Future.get(),其實是調用FutureTask.get(),其中會判斷任務狀態尚未結束,將主線程加入waiters等待鏈表,并掛起主線程。待任務執行結束后,FutureTask會喚醒所有等待獲取返回值的線程,此時主線程的FutureTask.get()就會返回了。所以,主線程和運行線程是通過FutureTask作為橋梁獲取線程返回值的。

3、Future.cancel()真的能取消任務的執行嗎?

首先答案是“不一定”,根據JDK中的方法注釋“Attempts to cancel execution of this task”,即嘗試去取消執行的任務。如果任務正在執行,且調用cancel()時,參數mayInterruptIfRunning傳的是true,那么會對執行線程調用interrupt()方法。那么問題就變成了interrupt()方法能中斷線程執行嗎?interrupt()方法不會中斷正在運行的線程。這一方法實際上完成的是在線程受到阻塞時拋出一個中斷信號,這樣線程就得以退出阻塞的狀態。更確切的說,如果線程被Object.wait()、Thread.join()、Thread.sleep()等阻塞,那么它將接收到一個中斷異常(InterruptedException),從而提早地終結被阻塞狀態。
如果線程沒有被阻塞,調用interrupt()將不起作用,那么即使線程正在阻塞狀態,并拋出了InterruptedException,線程能否真的取消執行還要看代碼中是否捕獲了InterruptedException和有沒有做相應的對中斷標示的判斷邏輯。

感謝你能夠認真閱讀完這篇文章,希望小編分享的“Callable與Runnable有什么區別”這篇文章對大家有幫助,同時也希望大家多多支持億速云,關注億速云行業資訊頻道,更多相關知識等著你來學習!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

吕梁市| 祁连县| 通榆县| 岱山县| 常山县| 宁乡县| 扬州市| 调兵山市| 信丰县| 无棣县| 卢龙县| 佛教| 慈利县| 晋中市| 射洪县| 新民市| 呈贡县| 中阳县| 永安市| 绿春县| 大兴区| 仙居县| 广南县| 青海省| 石屏县| 靖边县| 镇原县| 景宁| 拜城县| 新营市| 台南市| 吉木萨尔县| 扎赉特旗| 鹰潭市| 五大连池市| 盖州市| 高平市| 志丹县| 林州市| 曲沃县| 松潘县|