Java使用Runnable和Callable實現多線程的區別詳解
使用Runnable和Callable接口實現多線程的區別
先看兩種實現方式的步驟:
1.實現Runnable接口
public class ThreadDemo{ public static void main(String[] args) { for (int i = 1; i <= 5; i++) { //創建並啟動由實現Runnable接口創建的線程 new Thread(new Runner(),"Thread"+i).start(); } } } //實現Runnable接口 class Runner implements Runnable{ //實現run方法 @Override public void run() { System.out.println(Thread.currentThread().getName()+"有實現Runnable接口創建"); } }
2.實現 Callable 接口
public class ThreadDemo{ public static void main(String[] args) throws ExecutionException, InterruptedException { for (int i = 1; i <= 5; i++) { //使用FutureTask保存線程結果 FutureTask<String> futureTask = new FutureTask<String>(new Caller()); //創建並啟動由實現Callable創建的線程 new Thread(futureTask,"Thread"+i).start(); //獲取線程執行結果 System.out.println(futureTask.get()); } } } //實現Callable接口 class Caller implements Callable{ //實現Call接口 @Override public Object call() throws Exception { String result = Thread.currentThread().getName()+"由實現Callable接口創建"; return result; } }
從以上代碼可以看出,使用 Callable 接口創建多線程時使用瞭 FutureTask 進行封裝,然後 FutureTask 作為參數傳給 Thread 的構造函數,而 FutureTask 的作用是存放 Callable 接口 call 方法的返回值。我們來看一下 FutureTask 的源碼
//FutureTask實現瞭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的構造函數 public FutureTask(Callable<V> callable) { if (callable == null) throw new NullPointerException(); this.callable = callable; this.state = NEW; }
從對 FutureTask 的源碼分析,我們可以看出 FutureTask 實現瞭 Runnable 接口的 run 方法,在 run 方法中調用瞭 Callable 的 call 方法,將結果保存到 result 中,通過 set 方法將結果存儲至 outcome 變量中。
通過以上分析,我們總結出使用 Runnable 和 Callable 接口的區別:
1.使用 Runnable 接口實現更加簡單,而 通過 Callable 接口創建線程需要 FutureTask 進行封裝;
2.通過實現 Runnable 接口創建的線程沒有返回值,而使用 Callable 接口創建的線程可以有返回結果,並保存在 FutureTask中;
3.通過實現 Runnable 接口創建線程不拋出異常,而使用 Callable 接口創建的線程會拋出異常;
從以上總結可以看出,我們也可以看出 Runnable 適用於無需返回值的場景,而 Callable 接口用於需要保存返回值的場景。
到此這篇關於Java使用Runnable和Callable實現多線程的區別詳解的文章就介紹到這瞭,更多相關Java Runnable Callable多線程內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- 解析Java異步之call future
- java面試常問的Runnable和Callable的區別
- Java多線程 Callable、Future 和FutureTask
- Java線程的三種創建方式
- Java並發教程之Callable和Future接口詳解