Java使用Runnable和Callable實(shí)現(xiàn)多線程的區(qū)別詳解
使用Runnable和Callable接口實(shí)現(xiàn)多線程的區(qū)別
先看兩種實(shí)現(xiàn)方式的步驟:
1.實(shí)現(xiàn)Runnable接口
public class ThreadDemo{ public static void main(String[] args) { for (int i = 1; i <= 5; i++) { //創(chuàng)建并啟動(dòng)由實(shí)現(xiàn)Runnable接口創(chuàng)建的線程 new Thread(new Runner(),"Thread"+i).start(); } } } //實(shí)現(xiàn)Runnable接口 class Runner implements Runnable{ //實(shí)現(xiàn)run方法 @Override public void run() { System.out.println(Thread.currentThread().getName()+"有實(shí)現(xiàn)Runnable接口創(chuàng)建"); } }
2.實(shí)現(xiàn) Callable 接口
public class ThreadDemo{ public static void main(String[] args) throws ExecutionException, InterruptedException { for (int i = 1; i <= 5; i++) { //使用FutureTask保存線程結(jié)果 FutureTask<String> futureTask = new FutureTask<String>(new Caller()); //創(chuàng)建并啟動(dòng)由實(shí)現(xiàn)Callable創(chuàng)建的線程 new Thread(futureTask,"Thread"+i).start(); //獲取線程執(zhí)行結(jié)果 System.out.println(futureTask.get()); } } } //實(shí)現(xiàn)Callable接口 class Caller implements Callable{ //實(shí)現(xiàn)Call接口 @Override public Object call() throws Exception { String result = Thread.currentThread().getName()+"由實(shí)現(xiàn)Callable接口創(chuàng)建"; return result; } }
從以上代碼可以看出,使用 Callable 接口創(chuàng)建多線程時(shí)使用了 FutureTask 進(jìn)行封裝,然后 FutureTask 作為參數(shù)傳給 Thread 的構(gòu)造函數(shù),而 FutureTask 的作用是存放 Callable 接口 call 方法的返回值。我們來看一下 FutureTask 的源碼
//FutureTask實(shí)現(xiàn)了RunnableFuture接口 public class FutureTask<V> implements RunnableFuture<V> {} //再看RunnableFuture接口,繼承了Runnable接口 public interface RunnableFuture<V> extends Runnable, Future<V> { void run(); } //回到FutureTask,找到run方法 public void run() { ... Callable<V> c = callable; if (c != null && state == NEW) { V result; boolean ran; try { //獲取call的返回值 result = c.call(); ran = true; } catch (Throwable ex) { result = null; ran = false; setException(ex); } if (ran) set(result); } ... } //看一下set方法 protected void set(V v) { if (UNSAFE.compareAndSwapInt(this, stateOffset, NEW, COMPLETING)) { outcome = v; ... } } //再看一下FutureTask的構(gòu)造函數(shù) public FutureTask(Callable<V> callable) { if (callable == null) throw new NullPointerException(); this.callable = callable; this.state = NEW; }
從對 FutureTask 的源碼分析,我們可以看出 FutureTask 實(shí)現(xiàn)了 Runnable 接口的 run 方法,在 run 方法中調(diào)用了 Callable 的 call 方法,將結(jié)果保存到 result 中,通過 set 方法將結(jié)果存儲(chǔ)至 outcome 變量中。
通過以上分析,我們總結(jié)出使用 Runnable 和 Callable 接口的區(qū)別:
1.使用 Runnable 接口實(shí)現(xiàn)更加簡單,而 通過 Callable 接口創(chuàng)建線程需要 FutureTask 進(jìn)行封裝;
2.通過實(shí)現(xiàn) Runnable 接口創(chuàng)建的線程沒有返回值,而使用 Callable 接口創(chuàng)建的線程可以有返回結(jié)果,并保存在 FutureTask中;
3.通過實(shí)現(xiàn) Runnable 接口創(chuàng)建線程不拋出異常,而使用 Callable 接口創(chuàng)建的線程會(huì)拋出異常;
從以上總結(jié)可以看出,我們也可以看出 Runnable 適用于無需返回值的場景,而 Callable 接口用于需要保存返回值的場景。
到此這篇關(guān)于Java使用Runnable和Callable實(shí)現(xiàn)多線程的區(qū)別詳解的文章就介紹到這了,更多相關(guān)Java Runnable Callable多線程內(nèi)容請搜索腳本之家以前的文章或繼續(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的區(qū)別
- Java并發(fā)教程之Callable和Future接口詳解
- Java中callable的實(shí)現(xiàn)原理
相關(guān)文章
Spring Boot實(shí)現(xiàn)對文件進(jìn)行壓縮下載功能
在Web應(yīng)用中,文件下載功能是一個(gè)常見的需求,特別是當(dāng)你需要提供用戶下載各種類型的文件時(shí),本文將演示如何使用Spring Boot框架來實(shí)現(xiàn)一個(gè)簡單而強(qiáng)大的文件下載功能,需要的朋友跟隨小編一起學(xué)習(xí)吧2023-09-09Jmeter參數(shù)化實(shí)現(xiàn)方法及應(yīng)用實(shí)例
這篇文章主要介紹了Jmeter參數(shù)化實(shí)現(xiàn)方法及應(yīng)用實(shí)例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-08-08Spring工廠方法創(chuàng)建(實(shí)例化)bean實(shí)例代碼
這篇文章主要介紹了Spring工廠方法創(chuàng)建bean實(shí)例代碼,具有一定借鑒價(jià)值,需要的朋友可以參考下2018-01-01Scala 操作Redis使用連接池工具類RedisUtil
這篇文章主要介紹了Scala 操作Redis使用連接池工具類RedisUtil,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-06-06詳解Java編寫并運(yùn)行spark應(yīng)用程序的方法
這篇文章主要介紹了詳解Java編寫并運(yùn)行spark應(yīng)用程序的方法,內(nèi)容詳細(xì),結(jié)合了作者實(shí)際工作中的問題進(jìn)行具體分析,具有一定參考價(jià)值。2017-09-09ZooKeeper入門教程二在單機(jī)和集群環(huán)境下的安裝搭建及使用
本文是ZooKeeper入門系列教程,涵蓋ZooKeeper的安裝使及單機(jī)集群環(huán)境搭建,通過實(shí)例和大量圖表,結(jié)合實(shí)戰(zhàn),幫助學(xué)習(xí)者理解和運(yùn)用,有需要的朋友可以借鑒參考下2022-01-01java實(shí)現(xiàn)ATM機(jī)系統(tǒng)(2.0版)
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)ATM機(jī)系統(tǒng)2.0版,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-03-03Java編程中快速排序算法的實(shí)現(xiàn)及相關(guān)算法優(yōu)化
這篇文章主要介紹了Java編程中快速排序算法的實(shí)現(xiàn)及相關(guān)算法優(yōu)化,快速排序算法的最差時(shí)間復(fù)雜度為(n^2),最優(yōu)時(shí)間復(fù)雜度為(n\log n),存在優(yōu)化的空間,需要的朋友可以參考下2016-05-05