Java多線程編程(一):創建並運行Java線程

Java線程類也是一個Object類,它的實例都繼承自java.lang.Thread或其子類。可以用如下方式在Java中創建一個線程:

Thread thread = new Thread();

執行該線程可以調用該線程的start()方法:

thread.start();

在上面的例子中,我們並沒有爲線程編寫運行代碼,因此調用該方法後線程就終止了。
編寫線程運行時執行的代碼有兩種方式:
+一種是創建Thread子類的一個實例並重寫run方法
+第二種是創建類的時候實現Runnable接口。

創建Thread的子類

創建Thread子類的一個實例並重寫run方法,run方法會在調用start()方法之後被執行。

public class MyThread extends Thread {
   public void run(){
     System.out.println("MyThread running");
   }
}  

可以用如下方式創建並運行上述Thread子類:

MyThread myThread = new MyThread();
myThread.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方法。
同樣,也可以創建一個實現了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()方法,如下所示:

Thread newThread = new Thread(new MyRunnable());
newThread.run();

雖然也調用了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和操作系統一起決定了線程的執行順序,他和線程的啓動順序並非一定是一致的。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章