Java中實現執行緒的方式

CodingDiary發表於2019-07-11

Java中實現執行緒的方式

Java中實現多執行緒的方式的方式中最核心的就是 run()方法,不管何種方式其最終都是通過run()來執行。

Java剛釋出時也就是JDK 1.0版本提供了兩種實現方式,一個是繼承Thread類,一個是實現Runnable介面。兩種方式都是去重寫run()方法,在run()方法中去實現具體的業務程式碼。

但這兩種方式有一個共同的弊端,就是由於run()方法是沒有返回值的,所以通過這兩方式實現的多執行緒讀無法獲得執行的結果。

為了解決這個問題在JDK 1.5的時候引入一個Callable<V>介面,根據泛型V設定返回值的型別,實現他的call()方法,可以獲得執行緒執行的返回結果。

雖然call()方法可以獲得返回值,但它需要配合一個Future<V>才能拿到返回結果,而這個Future<V>又是繼承了Runnable的一個介面。 通過查閱原始碼就可以發現Future<V>的實現FutureTask<V>其在做具體業務程式碼執行的時候仍是在run()裡面實現的。

FutureTask 原始碼片段:

public void run() {
        if (state != NEW ||
            !UNSAFE.compareAndSwapObject(this, runnerOffset,
                                         null, Thread.currentThread()))
            return;
        try {
            Callable<V> c = callable;
            if (c != null && state == NEW) {
                V result;
                boolean ran;
                try {
                    result = c.call();
                    ran = true;
                } catch (Throwable ex) {
                    result = null;
                    ran = false;
                    setException(ex);
                }
                if (ran)
                    set(result);
            }
        } finally {
            // runner must be non-null until state is settled to
            // prevent concurrent calls to run()
            runner = null;
            // state must be re-read after nulling runner to prevent
            // leaked interrupts
            int s = state;
            if (s >= INTERRUPTING)
                handlePossibleCancellationInterrupt(s);
        }
    }

Java多執行緒實現方式的程式碼示例:

通過繼承Thread類實現

public class ThreadTest {

  public static void main(String[] args) throws Exception {
    Thread myThread = new MyThread();
    myThread.setName("MyThread-entends-Thread-test");
    myThread.start();

  }
}

class MyThread extends Thread {

  @Override
  public void run() {
    System.out.println("Thread Name:" + Thread.currentThread().getName());
  }
}

通過實現Runnable介面實現

public class ThreadTest {

  public static void main(String[] args) throws Exception {

    MyRunnableThread myRunnable = new MyRunnableThread();
    Thread myRunnableThread = new Thread(myRunnable);
    myRunnableThread.setName("MyThread-implements-Runnable-test");
    myRunnableThread.start();
  }
}

class MyRunnableThread implements Runnable {

  @Override
  public void run() {
    System.out.println("Thread Name:" + Thread.currentThread().getName());
  }
}

通過實現Callable介面實現

public class ThreadTest {

  public static void main(String[] args) throws Exception {

    Callable<String> myCallable = new MyCallableThread();
    FutureTask<String> futureTask = new FutureTask<>(myCallable);
    Thread myCallableThread = new Thread(futureTask);
    myCallableThread.setName("MyThread-implements-Callable-test");
    myCallableThread.start();
    System.out.println("Run by Thread:" + futureTask.get());

    //通過執行緒池執行
    ExecutorService executorService = Executors.newCachedThreadPool();
    executorService.submit(futureTask);
    executorService.shutdown();
    System.out.println("Run by ExecutorService:" + futureTask.get());
  }
}

class MyCallableThread implements Callable<String> {

  @Override
  public String call() throws Exception {
    return Thread.currentThread().getName();
  }
}

當然由於執行緒的建立和銷燬需要消耗資源,Java中還通過了許多執行緒池相關的API,上述示例中ExecutorService就是執行緒池API中的一個,關於執行緒池的詳細內容將會在下一篇繼續,歡迎大家關注。

相關文章