為什么需要定一個(gè)類去實(shí)現(xiàn)Runnable接口呢?繼承Thread類和實(shí)現(xiàn)Runnable接口有啥區(qū)別呢?
實(shí)現(xiàn)Runnable接口,避免了繼承Thread類的單繼承局限性。覆蓋Runnable接口中的run方法,將線程任務(wù)代碼定義到run方法中。
創(chuàng)建Thread類的對(duì)象,只有創(chuàng)建Thread類的對(duì)象才可以創(chuàng)建線程。線程任務(wù)已被封裝到Runnable接口的run方法中,而這個(gè)run方法所屬于Runnable接口的子類對(duì)象,所以將這個(gè)子類對(duì)象作為參數(shù)傳遞給Thread的構(gòu)造函數(shù),這樣,線程對(duì)象創(chuàng)建時(shí)就可以明確要運(yùn)行的線程的任務(wù)。
run()
線程對(duì)象調(diào)用run方法不開啟線程。僅是對(duì)象調(diào)用方法。
start()
線程對(duì)象調(diào)用start開啟線程,并讓jvm調(diào)用run方法在開啟的線程中執(zhí)行。
所以如果僅僅是調(diào)用run方法的話,就相當(dāng)于還是單線程,會(huì)順序執(zhí)行。
但是在Test類實(shí)現(xiàn)Runnable接口之后,Test類是沒有start()方法的,只有run()方法。這時(shí)調(diào)用run方法也僅僅是調(diào)用一個(gè)普通方法,不會(huì)開啟新線程。
我們此時(shí)就需要Thread類的start()方法來幫我們實(shí)現(xiàn)我們的多線程任務(wù)。
下面看代碼:
自定義線程執(zhí)行任務(wù)類
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
|
public class MyRunnable implements Runnable{ //定義線程要執(zhí)行的run方法邏輯 @Override public void run() { for ( int i = 0 ; i < 10 ; i++) { System.out.println( "我的線程:正在執(zhí)行!" +i); } } } public class Demo02 { public static void main(String[] args) { //創(chuàng)建線程執(zhí)行目標(biāo)類對(duì)象 Runnable runn = new MyRunnable(); //將Runnable接口的子類對(duì)象作為參數(shù)傳遞給Thread類的構(gòu)造函數(shù) Thread thread = new Thread(runn); Thread thread2 = new Thread(runn); //開啟線程 thread.start(); thread2.start(); for ( int i = 0 ; i < 10 ; i++) { System.out.println( "main線程:正在執(zhí)行!" +i); } } } |
補(bǔ)充:線程的啟動(dòng)的兩種方法,Runnable接口,run()的調(diào)用
實(shí)現(xiàn)并啟動(dòng)線程有兩種方法
1、寫一個(gè)類繼承自Thread類,重寫run方法。用start方法啟動(dòng)線程
2、寫一個(gè)類實(shí)現(xiàn)Runnable接口,實(shí)現(xiàn)run方法。用new Thread(Runnable target).start()方法來啟動(dòng)
多線程原理:相當(dāng)于玩游戲機(jī),只有一個(gè)游戲機(jī)(cpu),可是有很多人要玩,于是,start是排隊(duì)!等CPU選中你就是輪到你,你就run(),當(dāng)CPU的運(yùn)行的時(shí)間片執(zhí)行完,這個(gè)線程就繼續(xù)排隊(duì),等待下一次的run()。
調(diào)用start()后,線程會(huì)被放到等待隊(duì)列,等待CPU調(diào)度,并不一定要馬上開始執(zhí)行,只是將這個(gè)線程置于可動(dòng)行狀態(tài)。然后通過JVM,線程Thread會(huì)調(diào)用run()方法,執(zhí)行本線程的線程體。先調(diào)用start后調(diào)用run,這么麻煩,為了不直接調(diào)用run?就是為了實(shí)現(xiàn)多線程的優(yōu)點(diǎn),沒這個(gè)start不行。
1.start()方法來啟動(dòng)線程,真正實(shí)現(xiàn)了多線程運(yùn)行。
這時(shí)無需等待run方法體代碼執(zhí)行完畢,可以直接繼續(xù)執(zhí)行下面的代碼;通過調(diào)用Thread類的start()方法來啟動(dòng)一個(gè)線程, 這時(shí)此線程是處于就緒狀態(tài), 并沒有運(yùn)行。 然后通過此Thread類調(diào)用方法run()來完成其運(yùn)行操作的, 這里方法run()稱為線程體,它包含了要執(zhí)行的這個(gè)線程的內(nèi)容, Run方法運(yùn)行結(jié)束, 此線程終止。然后CPU再調(diào)度其它線程。
2.run()方法當(dāng)作普通方法的方式調(diào)用。
程序還是要順序執(zhí)行,要等待run方法體執(zhí)行完畢后,才可繼續(xù)執(zhí)行下面的代碼; 程序中只有主線程——這一個(gè)線程, 其程序執(zhí)行路徑還是只有一條, 這樣就沒有達(dá)到寫線程的目的。
記住:多線程就是分時(shí)利用CPU,宏觀上讓所有線程一起執(zhí)行 ,也叫并發(fā)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持服務(wù)器之家。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教。
原文鏈接:https://blog.csdn.net/qq_33247435/article/details/100135504