Java 線程類也是一個 object 類,它的實例都繼承自 java.lang.Thread 或其子類。 可以用如下方式用 java 中創建一個線程,執行該線程可以調用該線程的 start()方法:
Tread thread = new Thread();
thread.start();
在上面的例子中,我們並沒有為線程編寫運行代碼,因此調用該方法後線程就終止了。
編寫線程運行時執行的代碼有兩種方式:一種是創建 Thread 子類的一個實例並重寫 run 方法,第二種是創建類的時候實現 Runnable 接口。接下來我們會具體講解這兩種方法:
創建 Thread 的子類
創建 Thread 子類的一個實例並重寫 run 方法,run 方法會在調用 start()方法之後被執行。可以用如下方式創建並運行上述 Thread 子類例子如下:
public class MyThread extends Thread {
public void run(){
System.out.println("MyThread running");
}
}
MyThread myThread = new MyThread();
myTread.start();
一旦線程啟動後 start 方法就會立即返回,而不會等待到 run 方法執行完畢才返回。就好像 run 方法是在另外一個 cpu 上執行一樣。當 run 方法執行後,將會打印出字符串 MyThread running。
你也可以如下創建一個 Thread 的匿名子類:
Thread thread = new Thread(){
public void run(){
System.out.println("Thread Running");
}
};
thread.start();
當新的線程的 run 方法執行以後,計算機將會打印出字符串”Thread Running”。
實現 Runnable 接口
第二種編寫線程執行代碼的方式是新建一個實現了 java.lang.Runnable 接口的類的實例,實例中的方法可以被線程調用。下面給出例子:
public class MyRunnable implements Runnable {
public void run(){
System.out.println("MyRunnable running");
}
}
為了使線程能夠執行 run()方法,需要在 Thread 類的構造函數中傳入 MyRunnable 的實例對象。示例如下:
Thread thread = new Thread(new MyRunnable());
thread.start();
當線程運行時,它將會調用實現了 Runnable 接口的 run 方法。上例中將會打印出”MyRunnable running”。
同樣,也可以創建一個實現了 Runnable 接口的匿名類,如下所示:
Runnable myRunnable = new Runnable(){
public void run(){
System.out.println("Runnable running");
}
}
Thread thread = new Thread(myRunnable);
thread.start();
創建子類還是實現 Runnable 接口?
對於這兩種方式哪種好並沒有一個確定的答案,它們都能滿足要求。就我個人意見,我更傾向於實現 Runnable 接口這種方法。因為線程池可以有效的管理實現了 Runnable 接口的線程,如果線程池滿了,新的線程就會排隊等候執行,直到線程池空閒出來為止。而如果線程是通過實現 Thread 子類實現的,這將會復雜一些。
有時我們要同時融合實現 Runnable 接口和 Thread 子類兩種方式。例如,實現了 Thread 子類的實例可以執行多個實現了 Runnable 接口的線程。一個典型的應用就是線程池。
常見錯誤:調用 run()方法而非 start()方法
創建並運行一個線程所犯的常見錯誤是調用線程的 run()方法而非 start()方法,如下所示:
Thread newThread = new Thread(MyRunnable());
newThread.run(); //should be start();
起初你並不會感覺到有什麼不妥,因為 run()方法的確如你所願的被調用了。但是,事實上,run()方法並非是由剛創建的新線程所執行的,而是被創建新線程的當前線程所執行了。也就是被執行上面兩行代碼的線程所執行的。想要讓創建的新線程執行 run()方法,必須調用新線程的 start 方法。
線程名
當創建一個線程的時候,可以給線程起一個名字。它有助於我們區分不同的線程。例如:如果有多個線程寫入 System.out,我們就能夠通過線程名容易的找出是哪個線程正在輸出。例子如下:
MyRunnable runnable = new MyRunnable();
Thread thread = new Thread(runnable, "New Thread");
thread.start();
System.out.println(thread.getName());
需要注意的是,因為 MyRunnable 並非 Thread 的子類,所以 MyRunnable 類並沒有 getName()方法。可以通過以下方式得到當前線程的引用:
Thread.currentThread();
String threadName = Thread.currentThread().getName();
線程代碼舉例:
這裡是一個小小的例子。首先輸出執行main()方法線程名字。這個線程 JVM 分配的。然後開啟 10 個線程,命名為 1~10。每個線程輸出自己的名字後就退出。
public class ThreadExample {
public static void main(String[] args){
System.out.println(Thread.currentThread().getName());
for(int i=0; i<10; i++){
new Thread("" + i){
public void run(){
System.out.println("Thread: " + getName() + "running");
}
}.start();
}
}
}
需要注意的是,盡管啟動線程的順序是有序的,但是執行的順序並非是有序的。也就是說,1 號線程並不一定是第一個將自己名字輸出到控制台的線程。這是因為線程是並行執行而非順序的。Jvm 和操作系統一起決定了線程的執行順序,他和線程的啟動順序並非一定是一致的。