創(chuàng)建并運(yùn)行一個(gè)java線程方法介紹
要解釋線程,就必須明白什么是進(jìn)程。
什么是進(jìn)程呢?
進(jìn)程是指運(yùn)行中的應(yīng)用程序,每個(gè)進(jìn)程都有自己獨(dú)立的地址空間(內(nèi)存空間),比如用戶點(diǎn)擊桌面的IE瀏覽器,就啟動(dòng)了一個(gè)進(jìn)程,操作系統(tǒng)就會(huì)為該進(jìn)程分配獨(dú)立的地址空間。當(dāng)用戶再次點(diǎn)擊左面的IE瀏覽器,又啟動(dòng)了一個(gè)進(jìn)程,操作系統(tǒng)將為新的進(jìn)程分配新的獨(dú)立的地址空間。目前操作系統(tǒng)都支持多進(jìn)程。
要點(diǎn):用戶每啟動(dòng)一個(gè)進(jìn)程,操作系統(tǒng)就會(huì)為該進(jìn)程分配一個(gè)獨(dú)立的內(nèi)存空間。
線程--概念
在明白進(jìn)程后,就比較容易理解線程的概念。
什么是線程呢?
是進(jìn)程中的一個(gè)實(shí)體,是被系統(tǒng)獨(dú)立調(diào)度和分派的基本單位,線程自己不擁有系統(tǒng)資源,只擁有一點(diǎn)在運(yùn)行中必不可少的資源,但它可與同屬一個(gè)進(jìn)程的其它線程共享進(jìn)程所擁有的全部資源。一個(gè)線程可以創(chuàng)建和撤消另一個(gè)線程,同一進(jìn)程中的多個(gè)線程之間可以并發(fā)執(zhí)行。線程有就緒、阻塞和運(yùn)行三種基本狀態(tài)。
Java線程類也是一個(gè)object類,它的實(shí)例都繼承自java.lang.Thread或其子類。 可以用如下方式用java中創(chuàng)建一個(gè)線程:
Tread thread = new Thread();
執(zhí)行該線程可以調(diào)用該線程的start()
方法:
thread.start();
在上面的例子中,我們并沒有為線程編寫運(yùn)行代碼,因此調(diào)用該方法后線程就終止了。
編寫線程運(yùn)行時(shí)執(zhí)行的代碼有兩種方式:一種是創(chuàng)建Thread子類的一個(gè)實(shí)例并重寫run方法,第二種是創(chuàng)建類的時(shí)候?qū)崿F(xiàn)Runnable接口。接下來我們會(huì)具體講解這兩種方法:
創(chuàng)建Thread的子類
創(chuàng)建Thread子類的一個(gè)實(shí)例并重寫run方法,run方法會(huì)在調(diào)用start()方法之后被執(zhí)行。例子如下:
public class MyThread extends Thread { public void run(){ System.out.println("MyThread running"); } }
可以用如下方式創(chuàng)建并運(yùn)行上述Thread子類
MyThread myThread = new MyThread(); myTread.start();
一旦線程啟動(dòng)后start方法就會(huì)立即返回,而不會(huì)等待到run方法執(zhí)行完畢才返回。就好像run方法是在另外一個(gè)cpu上執(zhí)行一樣。當(dāng)run方法執(zhí)行后,將會(huì)打印出字符串MyThread running。
你也可以如下創(chuàng)建一個(gè)Thread的匿名子類:
Thread thread = new Thread(){ public void run(){ System.out.println("Thread Running"); } }; thread.start();
當(dāng)新的線程的run方法執(zhí)行以后,計(jì)算機(jī)將會(huì)打印出字符串”Thread Running”。
實(shí)現(xiàn)Runnable接口
第二種編寫線程執(zhí)行代碼的方式是新建一個(gè)實(shí)現(xiàn)了java.lang.Runnable接口的類的實(shí)例,實(shí)例中的方法可以被線程調(diào)用。
下面給出例子:
public class MyRunnable implements Runnable { public void run(){ System.out.println("MyRunnable running"); } }
為了使線程能夠執(zhí)行run()方法,需要在Thread類的構(gòu)造函數(shù)中傳入 MyRunnable的實(shí)例對(duì)象。示例如下:
Thread thread = new Thread(new MyRunnable()); thread.start();
當(dāng)線程運(yùn)行時(shí),它將會(huì)調(diào)用實(shí)現(xiàn)了Runnable接口的run方法。上例中將會(huì)打印出”MyRunnable running”。
同樣,也可以創(chuàng)建一個(gè)實(shí)現(xiàn)了Runnable接口的匿名類,如下所示:
Runnable myRunnable = new Runnable(){ public void run(){ System.out.println("Runnable running"); } } Thread thread = new Thread(myRunnable); thread.start();
創(chuàng)建子類還是實(shí)現(xiàn)Runnable接口?
對(duì)于這兩種方式哪種好并沒有一個(gè)確定的答案,它們都能滿足要求。就我個(gè)人意見,我更傾向于實(shí)現(xiàn)Runnable接口這種方法。因?yàn)榫€程池可以有效的管理實(shí)現(xiàn)了Runnable接口的線程,如果線程池滿了,新的線程就會(huì)排隊(duì)等候執(zhí)行,直到線程池空閑出來為止。而如果線程是通過實(shí)現(xiàn)Thread子類實(shí)現(xiàn)的,這將會(huì)復(fù)雜一些。
有時(shí)我們要同時(shí)融合實(shí)現(xiàn)Runnable接口和Thread子類兩種方式。例如,實(shí)現(xiàn)了Thread子類的實(shí)例可以執(zhí)行多個(gè)實(shí)現(xiàn)了Runnable接口的線程。一個(gè)典型的應(yīng)用就是線程池。
常見錯(cuò)誤:調(diào)用run()方法而非start()方法
創(chuàng)建并運(yùn)行一個(gè)線程所犯的常見錯(cuò)誤是調(diào)用線程的run()方法而非start()方法,如下所示:
Thread newThread = new Thread(MyRunnable()); newThread.run(); //should be start();
起初你并不會(huì)感覺到有什么不妥,因?yàn)閞un()方法的確如你所愿的被調(diào)用了。但是,事實(shí)上,run()方法并非是由剛創(chuàng)建的新線程所執(zhí)行的,而是被創(chuàng)建新線程的當(dāng)前線程所執(zhí)行了。也就是被執(zhí)行上面兩行代碼的線程所執(zhí)行的。想要讓創(chuàng)建的新線程執(zhí)行run()方法,必須調(diào)用新線程的start方法。
線程名
當(dāng)創(chuàng)建一個(gè)線程的時(shí)候,可以給線程起一個(gè)名字。它有助于我們區(qū)分不同的線程。例如:如果有多個(gè)線程寫入System.out,我們就能夠通過線程名容易的找出是哪個(gè)線程正在輸出。例子如下:
MyRunnable runnable = new MyRunnable(); Thread thread = new Thread(runnable, "New Thread"); thread.start(); System.out.println(thread.getName());
需要注意的是,因?yàn)镸yRunnable并非Thread的子類,所以MyRunnable類并沒有g(shù)etName()方法??梢酝ㄟ^以下方式得到當(dāng)前線程的引用:
Thread.currentThread();
因此,通過如下代碼可以得到當(dāng)前線程的名字:
String threadName = Thread.currentThread().getName();
線程代碼舉例:
這里是一個(gè)小小的例子。首先輸出執(zhí)行main()方法線程名字。這個(gè)線程JVM分配的。然后開啟10個(gè)線程,命名為1~10。每個(gè)線程輸出自己的名字后就退出。
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)行一個(gè)java線程方法介紹的全部內(nèi)容,希望對(duì)大家有所幫助。感興趣的朋友可以繼續(xù)參閱本站:淺談Java多線程的優(yōu)點(diǎn)及代碼示例、Java創(chuàng)建與結(jié)束線程代碼示例、Java線程安全的計(jì)數(shù)器簡單實(shí)現(xiàn)代碼示例等,如有不足之處,歡迎留言指出。感謝朋友們對(duì)本站的支持!
相關(guān)文章
Java反射機(jī)制的學(xué)習(xí)總結(jié)
總的來說,java反射機(jī)制是一個(gè)很好用的東西,用它可以解決很多死的東西,因?yàn)榉瓷錂C(jī)制的靈活行很大,有了他,我們就不要花太多的時(shí)間來寫操做數(shù)據(jù)庫的代碼了,而是方法更多的時(shí)間在項(xiàng)目的邏輯功能上,這個(gè)可以很大的減少開發(fā)時(shí)間,而且代碼的可讀性好2013-09-09SpringBoot可以同時(shí)處理多少請(qǐng)求流程分析
SpringBoot默認(rèn)的內(nèi)嵌容器是Tomcat,也就是我們的程序?qū)嶋H上是運(yùn)行在Tomcat里的,所以與其說SpringBoot可以處理多少請(qǐng)求,到不如說Tomcat可以處理多少請(qǐng)求,這篇文章主要介紹了SpringBoot可以同時(shí)處理多少請(qǐng)求,需要的朋友可以參考下2023-02-02javaweb實(shí)現(xiàn)投票系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了javaweb實(shí)現(xiàn)投票系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-09-09reactor-logback的AsyncAppender執(zhí)行流程源碼解讀
這篇文章主要為大家介紹了reactor-logback的AsyncAppender執(zhí)行流程源碼解讀,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-12-12SpringFox實(shí)現(xiàn)自動(dòng)生成RESTful?API文檔
在開發(fā)?RESTful?API?時(shí),編寫?API?文檔是一個(gè)重要的任務(wù),這篇文章為大家介紹了如何使用?SpringFox?自動(dòng)生成?RESTful?API?文檔,并提供示例代碼,需要的可以參考一下2023-06-06