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中的一個,關於執行緒池的詳細內容將會在下一篇繼續,歡迎大家關注。