這篇文章主要介紹了Callable與Runnable有什么區(qū)別,具有一定借鑒價(jià)值,感興趣的朋友可以參考下,希望大家閱讀完這篇文章之后大有收獲,下面讓小編帶著大家一起了解一下。
創(chuàng)新互聯(lián)公司是專業(yè)的佛山網(wǎng)站建設(shè)公司,佛山接單;提供網(wǎng)站設(shè)計(jì)、網(wǎng)站建設(shè),網(wǎng)頁設(shè)計(jì),網(wǎng)站設(shè)計(jì),建網(wǎng)站,PHP網(wǎng)站建設(shè)等專業(yè)做網(wǎng)站服務(wù);采用PHP框架,可快速的進(jìn)行佛山網(wǎng)站開發(fā)網(wǎng)頁制作和功能擴(kuò)展;專業(yè)做搜索引擎喜愛的網(wǎng)站,專業(yè)的做網(wǎng)站團(tuán)隊(duì),希望更多企業(yè)前來合作!
java.lang.Runnable是一個(gè)接口,只有一個(gè)run()方法
public interface Runnable { public abstract void run(); }
run()方法
的返回值是void,故在執(zhí)行完任務(wù)后無法返回任何結(jié)果
Callable是java.util.concurrent包下的,也是一個(gè)接口,也只有一個(gè)call()方法
,類似于java.lang.Runnable的run()方法
,實(shí)現(xiàn)Callable接口的類和實(shí)現(xiàn)Runnable接口的類都是可以被其它線程執(zhí)行的任務(wù)
public interface Callable{ 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()時(shí)被主線程捕獲;Runnable中的run()方法沒有定義拋出異常,運(yùn)行任務(wù)時(shí)發(fā)生異常時(shí)也會(huì)上拋,因?yàn)榧词共患幽J(rèn)也會(huì)上拋RuntimeException,但異常無法被主線程獲取
4、運(yùn)行Callable任務(wù)可以拿到一個(gè)Future對象代表異步運(yùn)算的結(jié)果
1、有了Runnable,為什么還需要Callable,它們的區(qū)別是什么?
Runnable和Callable都表示執(zhí)行的任務(wù),但不同的是Runnable.run()方法沒有返回值,Callable.call()有返回值
但其實(shí)線程在執(zhí)行任務(wù)時(shí)還是執(zhí)行的Runnable.run()方法,所以在使用ThreadPoolExecutor.submit()時(shí)會(huì)將Callable封裝為FutureTask,而FutureTask是Runnable和Future的實(shí)現(xiàn)類,所以在執(zhí)行Callable的任務(wù)時(shí),線程其實(shí)是執(zhí)行FutureTask這個(gè)Runnable的run()方法,其中封裝了調(diào)用Callable.call()并返回結(jié)果的邏輯。執(zhí)行Runnable任務(wù)如果發(fā)生異常,主線程無法知曉;而執(zhí)行Callable任務(wù)如果發(fā)生異常,在Future.get()時(shí)會(huì)拋出java.util.concurrent.ExecutionException,其中封裝了真實(shí)異常
2、Future.get()是如何獲取線程返回值的?
首先得益于Callable.call()方法定義了返回值,提交Callable任務(wù)后,Callable會(huì)被封裝成FutureTask,其既可以作為Runnable被執(zhí)行,也可以作為Future獲取返回值,F(xiàn)utureTask.run()方法會(huì)調(diào)用Callable.call()中的任務(wù)代碼。在任務(wù)執(zhí)行完成前,如果主線程使用Future.get(),其實(shí)是調(diào)用FutureTask.get(),其中會(huì)判斷任務(wù)狀態(tài)尚未結(jié)束,將主線程加入waiters等待鏈表,并掛起主線程。待任務(wù)執(zhí)行結(jié)束后,F(xiàn)utureTask會(huì)喚醒所有等待獲取返回值的線程,此時(shí)主線程的FutureTask.get()就會(huì)返回了。所以,主線程和運(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í),參數(shù)mayInterruptIfRunning傳的是true,那么會(huì)對執(zhí)行線程調(diào)用interrupt()方法。那么問題就變成了interrupt()方法能中斷線程執(zhí)行嗎?interrupt()方法不會(huì)中斷正在運(yùn)行的線程。這一方法實(shí)際上完成的是在線程受到阻塞時(shí)拋出一個(gè)中斷信號(hào),這樣線程就得以退出阻塞的狀態(tài)。更確切的說,如果線程被Object.wait()、Thread.join()、Thread.sleep()等阻塞,那么它將接收到一個(gè)中斷異常(InterruptedException),從而提早地終結(jié)被阻塞狀態(tài)。
如果線程沒有被阻塞,調(diào)用interrupt()將不起作用,那么即使線程正在阻塞狀態(tài),并拋出了InterruptedException,線程能否真的取消執(zhí)行還要看代碼中是否捕獲了InterruptedException和有沒有做相應(yīng)的對中斷標(biāo)示的判斷邏輯。
感謝你能夠認(rèn)真閱讀完這篇文章,希望小編分享的“Callable與Runnable有什么區(qū)別”這篇文章對大家有幫助,同時(shí)也希望大家多多支持創(chuàng)新互聯(lián),關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,更多相關(guān)知識(shí)等著你來學(xué)習(xí)!