如何建立並執行java執行緒
原文連結 譯者:章筱虎 校對:方騰飛
Java執行緒類也是一個object類,它的例項都繼承自java.lang.Thread或其子類。 可以用如下方式用java中建立一個執行緒:
Tread 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(); 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和作業系統一起決定了執行緒的執行順序,他和執行緒的啟動順序並非一定是一致的。