摘要:例子如下可以用如下方式創(chuàng)建并運(yùn)行上述子類一旦線程啟動后方法就會立即返回,而不會等待到方法執(zhí)行完畢才返回。但是,事實(shí)上方法并非是由剛創(chuàng)建的新線程所執(zhí)行的,而是被創(chuàng)建新線程的當(dāng)前線程所執(zhí)行了。這是因?yàn)榫€程是并行執(zhí)行而非順序的。
Java線程類也是一個object類,它的實(shí)例都繼承自java.lang.Thread或其子類。?可以用如下方式用java中創(chuàng)建一個線程:
Tread thread = new Thread();
執(zhí)行該線程可以調(diào)用該線程的start()方法:
thread.start();
在上面的例子中,我們并沒有為線程編寫運(yùn)行代碼,因此調(diào)用該方法后線程就終止了。
編寫線程運(yùn)行時執(zhí)行的代碼有兩種方式:一種是創(chuàng)建Thread子類的一個實(shí)例并重寫run方法,第二種是創(chuàng)建類的時候?qū)崿F(xiàn)Runnable接口。接下來我們會具體講解這兩種方法:
創(chuàng)建Thread的子類創(chuàng)建Thread子類的一個實(shí)例并重寫run方法,run方法會在調(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();
一旦線程啟動后start方法就會立即返回,而不會等待到run方法執(zhí)行完畢才返回。就好像run方法是在另外一個cpu上執(zhí)行一樣。當(dāng)run方法執(zhí)行后,將會打印出字符串MyThread running。
你也可以如下創(chuàng)建一個Thread的匿名子類:
Thread thread = new Thread(){ public void run(){ System.out.println("Thread Running"); } }; thread.start();
當(dāng)新的線程的run方法執(zhí)行以后,計(jì)算機(jī)將會打印出字符串”Thread Running”。
實(shí)現(xiàn)Runnable接口第二種編寫線程執(zhí)行代碼的方式是新建一個實(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í)例對象。示例如下:
Thread thread = new Thread(new MyRunnable()); thread.start();
當(dāng)線程運(yùn)行時,它將會調(diào)用實(shí)現(xiàn)了Runnable接口的run方法。上例中將會打印出”MyRunnable running”。
同樣,也可以創(chuàng)建一個實(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接口?
對于這兩種方式哪種好并沒有一個確定的答案,它們都能滿足要求。就我個人意見,我更傾向于實(shí)現(xiàn)Runnable接口這種方法。因?yàn)榫€程池可以有效的管理實(shí)現(xiàn)了Runnable接口的線程,如果線程池滿了,新的線程就會排隊(duì)等候執(zhí)行,直到線程池空閑出來為止。而如果線程是通過實(shí)現(xiàn)Thread子類實(shí)現(xiàn)的,這將會復(fù)雜一些。
有時我們要同時融合實(shí)現(xiàn)Runnable接口和Thread子類兩種方式。例如,實(shí)現(xiàn)了Thread子類的實(shí)例可以執(zhí)行多個實(shí)現(xiàn)了Runnable接口的線程。一個典型的應(yīng)用就是線程池。
常見錯誤:調(diào)用run()方法而非start()方法創(chuàng)建并運(yùn)行一個線程所犯的常見錯誤是調(diào)用線程的run()方法而非start()方法,如下所示:
Thread newThread = new Thread(MyRunnable()); newThread.run();? //should be start();
起初你并不會感覺到有什么不妥,因?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)建一個線程的時候,可以給線程起一個名字。它有助于我們區(qū)分不同的線程。例如:如果有多個線程寫入System.out,我們就能夠通過線程名容易的找出是哪個線程正在輸出。例子如下:
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();線程代碼舉例:
這里是一個小小的例子。首先輸出執(zhí)行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(); } } }
需要注意的是,盡管啟動線程的順序是有序的,但是執(zhí)行的順序并非是有序的。也就是說,1號線程并不一定是第一個將自己名字輸出到控制臺的線程。這是因?yàn)榫€程是并行執(zhí)行而非順序的。Jvm和操作系統(tǒng)一起決定了線程的執(zhí)行順序,他和線程的啟動順序并非一定是一致的。
原文 Creating and Starting Java Threads?
譯者 章筱虎 ?校對 方騰飛
via ifeve
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/64047.html
摘要:多線程和并發(fā)問題是技術(shù)面試中面試官比較喜歡問的問題之一。線程可以被稱為輕量級進(jìn)程。一個守護(hù)線程是在后臺執(zhí)行并且不會阻止終止的線程。其他的線程狀態(tài)還有,和。上下文切換是多任務(wù)操作系統(tǒng)和多線程環(huán)境的基本特征。 多線程和并發(fā)問題是 Java 技術(shù)面試中面試官比較喜歡問的問題之一。在這里,從面試的角度列出了大部分重要的問題,但是你仍然應(yīng)該牢固的掌握J(rèn)ava多線程基礎(chǔ)知識來對應(yīng)日后碰到的問題。(...
摘要:線程可以被稱為輕量級進(jìn)程。一個守護(hù)線程是在后臺執(zhí)行并且不會阻止終止的線程。其他的線程狀態(tài)還有,和。上下文切換是多任務(wù)操作系統(tǒng)和多線程環(huán)境的基本特征。在的線程中并沒有可供任何對象使用的鎖和同步器。 原文:Java Multi-Threading and Concurrency Interview Questions with Answers 翻譯:并發(fā)編程網(wǎng) - 鄭旭東 校對:方騰飛 多...
摘要:內(nèi)存模型指定了如何與計(jì)算機(jī)內(nèi)存協(xié)同工作。內(nèi)部的內(nèi)存模型內(nèi)存模型在內(nèi)部使用,將內(nèi)存分為了線程棧和堆。下面的圖從邏輯角度給出了內(nèi)存模型每個運(yùn)行在內(nèi)部的線程都有自己的線程棧。部分線程棧和堆可能在某些時候會占用緩存和內(nèi)部寄存器。 Java內(nèi)存模型指定了JVM如何與計(jì)算機(jī)內(nèi)存協(xié)同工作。JVM是整個計(jì)算機(jī)的模型因此這個模型包含了內(nèi)存模型,也就是Java內(nèi)存模型。 如果你像要設(shè)計(jì)正確行為的并發(fā)程序,...
摘要:下面是線程相關(guān)的熱門面試題,你可以用它來好好準(zhǔn)備面試。線程安全問題都是由全局變量及靜態(tài)變量引起的。持有自旋鎖的線程在之前應(yīng)該釋放自旋鎖以便其它線程可以獲得自旋鎖。 最近看到網(wǎng)上流傳著,各種面試經(jīng)驗(yàn)及面試題,往往都是一大堆技術(shù)題目貼上去,而沒有答案。 不管你是新程序員還是老手,你一定在面試中遇到過有關(guān)線程的問題。Java語言一個重要的特點(diǎn)就是內(nèi)置了對并發(fā)的支持,讓Java大受企業(yè)和程序員...
摘要:如果沒有,那必須先執(zhí)行相應(yīng)的類加載過程。分配內(nèi)存在類加載檢查通過后,接下來虛擬機(jī)將為新生對象分配內(nèi)存。程序計(jì)數(shù)器主要有兩個作用字節(jié)碼解釋器通過改變程序計(jì)數(shù)器來依次讀取指令,從而實(shí)現(xiàn)代碼的流程控制,如順序執(zhí)行選擇循環(huán)異常處理。 目錄介紹 01.Java對象的創(chuàng)建過程 1.0 看下創(chuàng)建類加載過程 1.1 對象的創(chuàng)建 1.2 對象的內(nèi)存布局 02.Java內(nèi)存區(qū)域 2.0 運(yùn)行...
閱讀 3118·2021-11-18 10:02
閱讀 2627·2021-10-13 09:47
閱讀 3073·2021-09-22 15:07
閱讀 805·2019-08-30 15:43
閱讀 1821·2019-08-30 10:59
閱讀 1702·2019-08-29 15:34
閱讀 1713·2019-08-29 15:06
閱讀 453·2019-08-29 13:28