Java中Thread 和 Runnable的區(qū)別_第1頁
Java中Thread 和 Runnable的區(qū)別_第2頁
Java中Thread 和 Runnable的區(qū)別_第3頁
Java中Thread 和 Runnable的區(qū)別_第4頁
Java中Thread 和 Runnable的區(qū)別_第5頁
已閱讀5頁,還剩3頁未讀 繼續(xù)免費(fèi)閱讀

下載本文檔

版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請(qǐng)進(jìn)行舉報(bào)或認(rèn)領(lǐng)

文檔簡(jiǎn)介

1、Thread and RunnableThread is the 進(jìn)程, Runnable is the 進(jìn)程對(duì)象第一需要弄清的問題如同程序和進(jìn)程的區(qū)別,要掌握多線程編程,第一要弄清的問題是:線程對(duì)象和線程的區(qū)別。線程對(duì)象是可以產(chǎn)生線程的對(duì)象。比如在java平臺(tái)中Thread對(duì)象,Runnable對(duì)象。線程,是指正在執(zhí)行的一個(gè)指令序列。在java平臺(tái)上是指從一個(gè)線程對(duì)象的start()開始,運(yùn)行run方法體中的那一段相對(duì)獨(dú)立的過程。鑒于作者的水平,無法用更確切的詞匯來描述它們的定義。但這兩個(gè)有本質(zhì)區(qū)別的概念請(qǐng)初學(xué)者細(xì)細(xì)體會(huì),隨著介紹的深入和例程分析的增加,就會(huì)慢慢明白它們所代表的真實(shí)含義。天下

2、難事必始于易,天下大事必始于細(xì)。讓我們先從最簡(jiǎn)單的"單線程"來入手:(1)帶引號(hào)說明只是相對(duì)而言的單線程,(2)基于java。 class BeginClasspublic static void main(String args)for(int i=0;i<100;i+)System.out.println("Hello,World!");如果我們成功編譯了該java文件,然后在命令行上敲入:java BeginClass現(xiàn)在發(fā)生了什么呢?每一個(gè)java程序員,從他開始學(xué)習(xí)java的第一分鐘里都會(huì)接觸到這個(gè)問題,但是,你知道它到底發(fā)生發(fā)

3、什么?JVM進(jìn)程被啟動(dòng),在同一個(gè)JVM進(jìn)程中,有且只有一個(gè)進(jìn)程,就是它自己。然后在這個(gè)JVM環(huán)境中,所有程序的運(yùn)行都是以線程來運(yùn)行。JVM最先會(huì)產(chǎn)生 一個(gè)主線程,由它來運(yùn)行指定程序的入口點(diǎn)。在這個(gè)程序中,就是主線程從main方法開始運(yùn)行。當(dāng)main方法結(jié)束后,主線程運(yùn)行完成。JVM進(jìn)程也隨之退出。我們看到的是一個(gè)主線程在運(yùn)行main方法,這樣的只有一個(gè)線程執(zhí)行程序邏輯的流程我們稱之為單線程。這是JVM提供給我們的單線程環(huán)境,事實(shí)上,JVM底層還至少有垃圾回收這樣的后臺(tái)線程以及其它非java線程,但這些線程對(duì)我們而言不可訪問,我們只認(rèn)為它是單線程的。主線程是JVM自己?jiǎn)?dòng)的,在這里它不是從線程對(duì)

4、象產(chǎn)生的。在這個(gè)線程中,它運(yùn)行了main方法這個(gè)指令序列。理解它,但它沒有更多可以研究的內(nèi)容。接觸多線程 class MyThread extends Threadpublic void run()System.out.println("Thread say:Hello,World!");public class MoreThreadspublic static void main(String args)new MyThread();new MyThread().start();System.out.println("Main say:Hello,World&q

5、uot;);執(zhí)行這個(gè)程序,main方法第一行產(chǎn)生了一個(gè)線程對(duì)象,但并沒有線程啟動(dòng)。main方法第二行產(chǎn)生了一個(gè)線程對(duì)象,并啟動(dòng)了一個(gè)線程。main方法第三行,產(chǎn)生并啟動(dòng)一個(gè)線程后,主線程自己也繼續(xù)執(zhí)行其它語句。我們先不研究Thread對(duì)象的具體內(nèi)容,稍微來回想一下上面的兩個(gè)概念,線程對(duì)象和線程。在JAVA中,線程對(duì)象是JVM產(chǎn)生的一個(gè)普通的Object子類。而線程是CPU分配給這個(gè)對(duì)象的一個(gè)運(yùn)行過程。我們說的這個(gè)線程在干什么,不是說一個(gè)線程對(duì)象在干什么,而是這個(gè)運(yùn)行過程在干什么。如果一時(shí)想不明白,不要急,但你要記得它們不是一回事就行了?,F(xiàn)在我們來開始考察JAVA中線程對(duì)象:在JAVA中,要開始

6、一個(gè)線程,有兩種方式。一是直接調(diào)用Thread實(shí)例的start()方法,二是將Runable實(shí)例傳給一個(gè)Thread實(shí)例然后調(diào)用它的start()方法。在前面已經(jīng)說過,線程對(duì)象和線程是兩個(gè)完全不同的概念。這里我們?cè)俅紊钊胍幌?,生成一個(gè)線程的實(shí)例,并不代表啟動(dòng)了線程。而啟動(dòng)線程是說在某個(gè)線程對(duì)象上啟動(dòng)了該實(shí)例對(duì)應(yīng)的線程,當(dāng)該線程結(jié)束后,并不會(huì)就立即消失。對(duì)于從很多書籍上可以看到的基礎(chǔ)知識(shí)我就不用多說了。既然是基礎(chǔ)知識(shí),我也著重于從普通文檔上讀不到的內(nèi)容。所以本節(jié)我重點(diǎn)要說的是兩種線程對(duì)象產(chǎn)生線程方式的區(qū)別。class MyThread extends Threadpublic int x = 0

7、;public void run()for(int i=0;i<100;i+)tryThread.sleep(10);catch(Exception e)System.out.println(x+);如果我們生成MyThread的一個(gè)實(shí)例,然后調(diào)用它的start()方法,那么就產(chǎn)生了這個(gè)實(shí)例對(duì)應(yīng)的線程:public class Test public static void main(String args) throws ExceptionMyThread mt = new MyThread();mt.start();不用說,最終會(huì)打印出0到99,現(xiàn)在我們稍微玩一點(diǎn)花樣:public

8、class Test public static void main(String args) throws ExceptionMyThread mt = new MyThread();mt.start();System.out.println(101);也不用說,在基礎(chǔ)篇(一)中我們知道由于單CPU的原因,一般會(huì)先打印101,然后打印0到99。不過我們可以控制線程讓它按我們的意思來運(yùn)行:public class Test public static void main(String args) throws ExceptionMyThread mt = new MyThread();mt.s

9、tart();mt.join();System.out.println(101);好了,我們終于看到,mt實(shí)例對(duì)應(yīng)的線程(假如我有時(shí)說mt線程請(qǐng)你不要怪我,不過我盡量不這么說)。在運(yùn)行完成后,主線程才打印101。因?yàn)?我們讓當(dāng)前線程(這里是主線程)等待mt線程的運(yùn)行結(jié)束。"在線程對(duì)象a上調(diào)用join()方法,就是讓當(dāng)前正在執(zhí)行的線程等待線程對(duì)象a對(duì)應(yīng)的線程運(yùn)行 完成后才繼續(xù)運(yùn)行。" 請(qǐng)大家一定要深刻理解并熟記這句話,而我這里引出這個(gè)知識(shí)點(diǎn)的目的是為了讓你繼續(xù)看下面的例子:public class Test public static void main(Strin

10、g args) throws ExceptionMyThread mt = new MyThread();mt.start();mt.join();Thread.sleep(3000);mt.start();當(dāng)線程對(duì)象mt運(yùn)行完成后,我們讓主線程休息一下,然后我們?cè)俅卧谶@個(gè)線程對(duì)象上啟動(dòng)線程。結(jié)果我們看到:Exception in thread "main" java.lang.IllegalThreadStateException也就是這種線程對(duì)象一時(shí)運(yùn)行一次完成后,它就再也不能運(yùn)行第二次了。我們可以看一下它有具體實(shí)現(xiàn): publi

11、c synchronized void start() if (started)throw new IllegalThreadStateException();started = true;group.add(this);start0();一個(gè)Thread的實(shí)例一旦調(diào)用start()方法,這個(gè)實(shí)例的started標(biāo)記就標(biāo)記為true,事實(shí)中不管這個(gè)線程后來有沒有執(zhí)行到底,只要調(diào)用了一次start()就再也沒有機(jī)會(huì)運(yùn)行了,這意味著:通過Thread實(shí)例的start(),一個(gè)Thread的實(shí)例只能產(chǎn)生一個(gè)線程那么如果要在一個(gè)實(shí)例上產(chǎn)生多個(gè)線程(也就是我們常說的線程池),我們應(yīng)該如何做呢?這就是Ru

12、nnable接口給我們帶來的偉大的功能。class R implements Runnableprivate int x = 0;public void run()for(int i=0;i<100;i+)tryThread.sleep(10);catch(Exception e)System.out.println(x+);正如它的名字一樣,Runnable的實(shí)例是可運(yùn)行的,但它自己并不能直接運(yùn)行,它需要被Thread對(duì)象來包裝才行運(yùn)行:public class Test public static void main(String args) throws Exceptionnew

13、Thread(new R().start();當(dāng)然這個(gè)結(jié)果和mt.start()沒有什么區(qū)別。但如果我們把一個(gè)Runnable實(shí)例給Thread對(duì)象多次包裝,我們就可以看到它們實(shí)際是在同一實(shí)例上啟動(dòng)線程:public class Test public static void main(String args) throws ExceptionR r = new R(); /Runnable對(duì)象只是創(chuàng)建了一次,但是通過for循環(huán)用Thread包裝后生成了十個(gè)Thread線程對(duì)象,并啟動(dòng)了10次,但是10次都是針對(duì)這同一個(gè)Runnable對(duì)象的,執(zhí)行的也是Runnable的同一個(gè)run函數(shù),同一

14、段代碼。for(int i=0;i<10;i+)new Thread(r).start();x是實(shí)例對(duì)象,但結(jié)果是x被加到了999,說明這10個(gè)線程是在同一個(gè)r對(duì)象上運(yùn)行的。請(qǐng)大家注意,因?yàn)檫@個(gè)例子是在單CPU上運(yùn)行的,所以沒有對(duì)多個(gè)線程同時(shí)操作共同的對(duì)象進(jìn)行同步。這里是為了說明的方便而簡(jiǎn)化了同步,而真正的環(huán)境中你無法預(yù)知程序會(huì)在什么環(huán)境下運(yùn)行,所以一定要考慮同步。到這里我們做一個(gè)完整的例子來說明線程產(chǎn)生的方式不同而生成的線程的區(qū)別:package debug;import java.io.*;import java.lang.Thread;class MyThread extends

15、Threadpublic int x = 0;public void run()System.out.println(+x);class R implements Runnableprivate int x = 0;public void run()System.out.println(+x);public class Test public static void main(String args) throws Exceptionfor(int i=0;i<10;i+)Thread t = new MyThread();t.start();Thread.sleep(10000);/讓

16、上面的線程運(yùn)行完成R r = new R();for(int i=0;i<10;i+)Thread t = new Thread(r);t.start();上面10個(gè)線程對(duì)象產(chǎn)生的10個(gè)線程運(yùn)行時(shí)打印了10次1。下面10個(gè)線程對(duì)象產(chǎn)生的10個(gè)線程運(yùn)行時(shí)打印了1到10。我們把下面的10個(gè)線程稱為同一實(shí)例(Runnable實(shí)例)的多個(gè)線程。package a.b;class MyThread extends Thread public int x = 0;/為了驗(yàn)證每次新建一個(gè)Thread對(duì)象時(shí),是否會(huì)執(zhí)行run方法之外的初始化代碼,這里加了一個(gè)非靜態(tài)代碼塊,因?yàn)槿舴庆o態(tài)代碼塊執(zhí)行了,那么p

17、ublic int x = 0也會(huì)被執(zhí)行。public int count = 1;System.out.println("in MyThread non-static block: x=" + count+);public void run() System.out.println("in MyThread run: x="+ +x);class R implements Runnable /為了驗(yàn)證每次新建一個(gè)Thread對(duì)象時(shí),是否會(huì)執(zhí)行run方法之外的初始化代碼,這里加了一個(gè)非靜態(tài)代碼塊,因?yàn)槿舴庆o態(tài)代碼塊執(zhí)行了,那么public int x =

18、 0也會(huì)被執(zhí)行。private int x = 0;System.out.println("in Runnable non-static block: x=" + x);public void run() System.out.println("in Runnable run : x=" + +x);public class ThreadTest public static void main(String args) throws Exception int x = 1;/R r = new R();for (int i = 0; i < 10; i+) /Thread t = new Thread(r);Thread t = new MyThread();t.start()

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請(qǐng)下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請(qǐng)聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會(huì)有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
  • 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
  • 5. 人人文庫網(wǎng)僅提供信息存儲(chǔ)空間,僅對(duì)用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對(duì)用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對(duì)任何下載內(nèi)容負(fù)責(zé)。
  • 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請(qǐng)與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時(shí)也不承擔(dān)用戶因使用這些下載資源對(duì)自己和他人造成任何形式的傷害或損失。

最新文檔

評(píng)論

0/150

提交評(píng)論