多線程_解決Runnable接口無start()方法的情況
為什么需要定一個(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ù)類
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è)參考,也希望大家多多支持腳本之家。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教。
相關(guān)文章
簡(jiǎn)單了解spring cloud 網(wǎng)關(guān)服務(wù)
這篇文章主要介紹了簡(jiǎn)單了解spring cloud 網(wǎng)關(guān)服務(wù),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-10-10
Java日常練習(xí)題,每天進(jìn)步一點(diǎn)點(diǎn)(51)
下面小編就為大家?guī)硪黄狫ava基礎(chǔ)的幾道練習(xí)題(分享)。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧,希望可以幫到你2021-08-08
Java 手動(dòng)解析不帶引號(hào)的JSON字符串的操作
這篇文章主要介紹了Java 手動(dòng)解析不帶引號(hào)的JSON字符串的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-10-10
Java實(shí)現(xiàn)微信登錄并獲取用戶信息功能(開發(fā)流程)
這篇文章主要介紹了Java實(shí)現(xiàn)微信登錄并獲取用戶信息功能(開發(fā)流程),本文通過實(shí)例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下2024-07-07
Java中數(shù)組容器(ArrayList)設(shè)的計(jì)與實(shí)現(xiàn)
本篇文章主要跟大家介紹我們最常使用的一種容器ArrayList、Vector的原理,并且自己使用Java實(shí)現(xiàn)自己的數(shù)組容器MyArrayList,讓自己寫的容器能像ArrayList那樣工作,感興趣的可以了解一下2022-07-07
java實(shí)戰(zhàn)案例之用戶注冊(cè)并發(fā)送郵件激活/發(fā)送郵件驗(yàn)證碼
現(xiàn)在很多的網(wǎng)站都提供有用戶注冊(cè)功能,當(dāng)我們注冊(cè)成功之后就會(huì)收到封注冊(cè)網(wǎng)站的郵件,郵件里包含了我們的注冊(cè)的用戶名和密碼及激活賬戶的超鏈接等信息,這篇文章主要給大家介紹了關(guān)于java實(shí)戰(zhàn)案例之用戶注冊(cè)并發(fā)送郵件激活/發(fā)送郵件驗(yàn)證碼的相關(guān)資料,需要的朋友可以參考下2021-09-09

