溫馨提示×

溫馨提示×

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

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

Callable與Runnable有什么區(qū)別

發(fā)布時間:2021-06-16 14:26:22 來源:億速云 閱讀:301 作者:小新 欄目:大數(shù)據(jù)

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

Callable 與 Runnable

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI