要解釋線程,就必須明白什么是進(jìn)程。
什么是進(jìn)程呢?
進(jìn)程是指運(yùn)行中的應(yīng)用程序,每個進(jìn)程都有自己獨立的地址空間(內(nèi)存空間),比如用戶點擊桌面的IE瀏覽器,就啟動了一個進(jìn)程,操作系統(tǒng)就會為該進(jìn)程分配獨立的地址空間。當(dāng)用戶再次點擊左面的IE瀏覽器,又啟動了一個進(jìn)程,操作系統(tǒng)將為新的進(jìn)程分配新的獨立的地址空間。目前操作系統(tǒng)都支持多進(jìn)程。
要點:用戶每啟動一個進(jìn)程,操作系統(tǒng)就會為該進(jìn)程分配一個獨立的內(nèi)存空間。
線程--概念
在明白進(jìn)程后,就比較容易理解線程的概念。
什么是線程呢?
是進(jìn)程中的一個實體,是被系統(tǒng)獨立調(diào)度和分派的基本單位,線程自己不擁有系統(tǒng)資源,只擁有一點在運(yùn)行中必不可少的資源,但它可與同屬一個進(jìn)程的其它線程共享進(jìn)程所擁有的全部資源。一個線程可以創(chuàng)建和撤消另一個線程,同一進(jìn)程中的多個線程之間可以并發(fā)執(zhí)行。線程有就緒、阻塞和運(yùn)行三種基本狀態(tài)。
Java線程類也是一個object類,它的實例都繼承自java.lang.Thread或其子類。 可以用如下方式用java中創(chuàng)建一個線程:
1
|
Tread thread = new Thread(); |
執(zhí)行該線程可以調(diào)用該線程的start()
方法:
1
|
thread.start(); |
在上面的例子中,我們并沒有為線程編寫運(yùn)行代碼,因此調(diào)用該方法后線程就終止了。
編寫線程運(yùn)行時執(zhí)行的代碼有兩種方式:一種是創(chuàng)建Thread子類的一個實例并重寫run方法,第二種是創(chuàng)建類的時候?qū)崿F(xiàn)Runnable接口。接下來我們會具體講解這兩種方法:
創(chuàng)建Thread的子類
創(chuàng)建Thread子類的一個實例并重寫run方法,run方法會在調(diào)用start()方法之后被執(zhí)行。例子如下:
1
2
3
4
5
|
public class MyThread extends Thread { public void run(){ System.out.println( "MyThread running" ); } } |
可以用如下方式創(chuàng)建并運(yùn)行上述Thread子類
1
2
|
MyThread myThread = new MyThread(); myTread.start(); |
一旦線程啟動后start方法就會立即返回,而不會等待到run方法執(zhí)行完畢才返回。就好像run方法是在另外一個cpu上執(zhí)行一樣。當(dāng)run方法執(zhí)行后,將會打印出字符串MyThread running。
你也可以如下創(chuàng)建一個Thread的匿名子類:
1
2
3
4
5
6
|
Thread thread = new Thread(){ public void run(){ System.out.println( "Thread Running" ); } }; thread.start(); |
當(dāng)新的線程的run方法執(zhí)行以后,計算機(jī)將會打印出字符串”Thread Running”。
實現(xiàn)Runnable接口
第二種編寫線程執(zhí)行代碼的方式是新建一個實現(xiàn)了java.lang.Runnable接口的類的實例,實例中的方法可以被線程調(diào)用。
下面給出例子:
1
2
3
4
5
|
public class MyRunnable implements Runnable { public void run(){ System.out.println( "MyRunnable running" ); } } |
為了使線程能夠執(zhí)行run()方法,需要在Thread類的構(gòu)造函數(shù)中傳入 MyRunnable的實例對象。示例如下:
1
2
|
Thread thread = new Thread( new MyRunnable()); thread.start(); |
當(dāng)線程運(yùn)行時,它將會調(diào)用實現(xiàn)了Runnable接口的run方法。上例中將會打印出”MyRunnable running”。
同樣,也可以創(chuàng)建一個實現(xiàn)了Runnable接口的匿名類,如下所示:
1
2
3
4
5
6
7
|
Runnable myRunnable = new Runnable(){ public void run(){ System.out.println( "Runnable running" ); } } Thread thread = new Thread(myRunnable); thread.start(); |
創(chuàng)建子類還是實現(xiàn)Runnable接口?
對于這兩種方式哪種好并沒有一個確定的答案,它們都能滿足要求。就我個人意見,我更傾向于實現(xiàn)Runnable接口這種方法。因為線程池可以有效的管理實現(xiàn)了Runnable接口的線程,如果線程池滿了,新的線程就會排隊等候執(zhí)行,直到線程池空閑出來為止。而如果線程是通過實現(xiàn)Thread子類實現(xiàn)的,這將會復(fù)雜一些。
有時我們要同時融合實現(xiàn)Runnable接口和Thread子類兩種方式。例如,實現(xiàn)了Thread子類的實例可以執(zhí)行多個實現(xiàn)了Runnable接口的線程。一個典型的應(yīng)用就是線程池。
常見錯誤:調(diào)用run()方法而非start()方法
創(chuàng)建并運(yùn)行一個線程所犯的常見錯誤是調(diào)用線程的run()方法而非start()方法,如下所示:
1
2
|
Thread newThread = new Thread(MyRunnable()); newThread.run(); //should be start(); |
起初你并不會感覺到有什么不妥,因為run()方法的確如你所愿的被調(diào)用了。但是,事實上,run()方法并非是由剛創(chuàng)建的新線程所執(zhí)行的,而是被創(chuàng)建新線程的當(dāng)前線程所執(zhí)行了。也就是被執(zhí)行上面兩行代碼的線程所執(zhí)行的。想要讓創(chuàng)建的新線程執(zhí)行run()方法,必須調(diào)用新線程的start方法。
線程名
當(dāng)創(chuàng)建一個線程的時候,可以給線程起一個名字。它有助于我們區(qū)分不同的線程。例如:如果有多個線程寫入System.out,我們就能夠通過線程名容易的找出是哪個線程正在輸出。例子如下:
1
2
3
4
|
MyRunnable runnable = new MyRunnable(); Thread thread = new Thread(runnable, "New Thread" ); thread.start(); System.out.println(thread.getName()); |
需要注意的是,因為MyRunnable并非Thread的子類,所以MyRunnable類并沒有g(shù)etName()方法。可以通過以下方式得到當(dāng)前線程的引用:
1
|
Thread.currentThread(); |
因此,通過如下代碼可以得到當(dāng)前線程的名字:
1
|
String threadName = Thread.currentThread().getName(); |
線程代碼舉例:
這里是一個小小的例子。首先輸出執(zhí)行main()方法線程名字。這個線程JVM分配的。然后開啟10個線程,命名為1~10。每個線程輸出自己的名字后就退出。
1
2
3
4
5
6
7
8
9
10
11
12
|
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(); } } } |
總結(jié)
以上就是本文關(guān)于創(chuàng)建并運(yùn)行一個java線程方法介紹的全部內(nèi)容,希望對大家有所幫助。如有不足之處,歡迎留言指出。感謝朋友們對本站的支持!
原文鏈接:http://www.importnew.com/20672.html