Java多線程Thread類的使用詳解
1.創(chuàng)建一個(gè)線程
Java操作線程最核心的類就是Thread類
創(chuàng)建線程有很多方法,下面我們寫一個(gè)Mythread類繼承 Thread 類重寫run()方法來(lái)創(chuàng)建線程
package thread; class MyThread extends Thread{ public void run(){ System.out.println("hello world!"); } } public class ThreaDemo1 { public static void main(String[] args) { Thread t = new MyThread(); } }
這里的Thread類不需要import包,是因?yàn)檫@個(gè)類和String,StringBuffer這些類一樣是在java.lang包中,已經(jīng)自動(dòng)導(dǎo)入了!
run()方法是重寫了Thread類的run()方法,重寫就是和父類方法名參數(shù)都相同,子類的方法通過(guò)動(dòng)態(tài)綁定機(jī)制被調(diào)用
就像上面的:
Thread t = new MyThread(); t.run();
這里的t是父類的引用,調(diào)用的run()仍然是子類的方法,本質(zhì)上t還是指向子類對(duì)象
重載是同一個(gè)作用域,多個(gè)方法的方法名相同,參數(shù)個(gè)數(shù)或類型不同的方法
2.start()方法與run()方法
我們用t調(diào)用一個(gè)strat方法
public class ThreaDemo1 { public static void main(String[] args) { Thread t = new MyThread(); t.start(); } }
這里的hello world和我們直接在打印出來(lái)的hello world是不同的!!!
這里的是由t.start()創(chuàng)建了一個(gè)新的線程,然后由線程負(fù)責(zé)執(zhí)行t.run(),打印出hello world!
t.start()創(chuàng)建新的線程時(shí),調(diào)用了操作系統(tǒng)的API,通過(guò)操作系統(tǒng)內(nèi)核創(chuàng)建新線程的PCB,并把要執(zhí)行的指令交給這個(gè)PCB,當(dāng)PCB被調(diào)度到CPU上執(zhí)行的時(shí)候,就能執(zhí)行到run()方法中的代碼了
直接打印hello world時(shí),java進(jìn)程就只有一個(gè)線程(就是調(diào)用main方法的線程)也就是主線程
通過(guò)t.start(),主線程調(diào)用了t.start(),t.start()創(chuàng)建出新的線程,新的線程調(diào)用t.run();
當(dāng)run()方法執(zhí)行完畢,新的的線程就自動(dòng)銷毀
創(chuàng)建線程歸結(jié)到底還是要解決并發(fā)編程問(wèn)題,我們來(lái)看一個(gè)并發(fā)的例子
class MyThread extends Thread{ public void run(){ while(true){ System.out.println("hello world!"); //休眠 try { Thread.sleep(1000); } catch (InterruptedException e) { throw new RuntimeException(e); } } } } public class ThreaDemo1 { public static void main(String[] args) { Thread t = new MyThread(); t.start(); while(true){ System.out.println("hello main!"); try { Thread.sleep(1000); } catch (InterruptedException e) { throw new RuntimeException(e); } } } }
這段代碼并不是只執(zhí)行到一個(gè)循環(huán),死循環(huán)出不來(lái)了,而是交替執(zhí)行的
這就是并發(fā)執(zhí)行的效果,執(zhí)行的順序也是不確定的,主要原因是:操作系統(tǒng)調(diào)度線程的時(shí)候是"搶占式執(zhí)行",具體哪個(gè)線程先執(zhí)行,哪個(gè)線程后執(zhí)行,都是不確定的,取決于操作系統(tǒng)調(diào)度策略
雖然也是有優(yōu)先級(jí)的,但是從應(yīng)用程序方面來(lái)看到的效果是隨機(jī)的,我們?cè)诔绦蛏蠠o(wú)法修改的
如果我們只執(zhí)行t.run()會(huì)是什么情況呢?
public static void main(String[ ] args)l Thread t = new MyThread(; t.run(); while(true){ system.out.println("hello main! "); try { Thread.sleep( millis: 1000) ; catch (InterruptedException e) { throw new RuntimeException(e); }
結(jié)果
這里只執(zhí)行了run方法,沒(méi)有執(zhí)行hellomain這里,是因?yàn)橹挥幸粋€(gè)主線程是單線程工作的
不會(huì)交替執(zhí)行,這就是單線程和多線程的差異
run()和start()的區(qū)別
run是線程里的方法,描述的是線程要執(zhí)行的任務(wù),但是線程還沒(méi)有執(zhí)行,只有調(diào)用start后,線程才會(huì)開(kāi)始運(yùn)行,調(diào)用后會(huì)申請(qǐng)線程并執(zhí)行run方法,執(zhí)行完后進(jìn)入銷毀線程階段,也就是說(shuō)strat才是真正創(chuàng)建了線程,并且線程是獨(dú)立的執(zhí)行流
3.查看線程
我們可以使用jdk自帶的工具jconsole查看當(dāng)前的java進(jìn)程中的所有線程
這個(gè)工具一般路徑:C:\Program Files\Java\jdk1.8.0_192\bin
雙擊打開(kāi)運(yùn)行會(huì)看到本地進(jìn)程
接下來(lái)點(diǎn)擊一下我們運(yùn)行的程序的線程
點(diǎn)擊線程
可以看到有很多線程
詳細(xì)信息
其它的線程都是JVM自帶的線程
注意:如果打開(kāi)工具本地進(jìn)程什么也沒(méi)有,可以嘗試用右鍵,管理員方式運(yùn)行
new Thread 對(duì)象時(shí),不創(chuàng)建線程,調(diào)用start才是創(chuàng)建PCB才有真實(shí)的線程
主線程調(diào)用就是start方法寫到main方法中被調(diào)用
4.創(chuàng)建線程的各種方法
Java中創(chuàng)建線程的方法有很多種,上面使用的就是繼承Thread類,重寫run()方法的方法,下面了解一下其他三種方法
4.1實(shí)現(xiàn)Runnable接口
回顧:
抽象類和普通類的區(qū)別:抽象類不能實(shí)例化,不能直接new,必須有子類繼承抽象類,然后抽象類中有抽象方法,抽象方法沒(méi)有方法體,需要子類重寫抽象方法
接口比抽象類更進(jìn)一步,抽象類除抽象方法之外還有普通方法和屬性,接口則是只有抽象方法
class MyRunnable implements Runnable{ public void run(){ System.out.println("hello thread!"); } } public class ThreadDemo2 { public static void main(String[] args) { Runnable runnable = new MyRunnable(); Thread t = new Thread(runnable); t.start(); } }
Runnable是描述任務(wù),具體執(zhí)行細(xì)節(jié)就是run()方法,任務(wù)還要交給線程來(lái)執(zhí)行
使用Runnable接口達(dá)到了解耦合的目的,將線程和線程的任務(wù)分開(kāi),如果要修改代碼,代碼改動(dòng)就比較小
4.2使用匿名內(nèi)部類
public class ThreadDemo3 { public static void main(String[] args) { Thread t = new Thread(){ @Override public void run() { System.out.println("hello world!"); } }; t.start(); } }
這里創(chuàng)建了一個(gè)Thread的子類,子類是匿名的,還創(chuàng)建了子類的實(shí)例,并且讓t指向?qū)嵗?/p>
4.3使用匿名內(nèi)部類實(shí)現(xiàn)Runnable
public class ThreadDemo4 { public static void main(String[] args) { Thread t = new Thread(new Runnable() { @Override public void run() { System.out.println("hello world!!"); } }); t.start(); } }
寫法和1本質(zhì)相同,只不過(guò)把實(shí)現(xiàn)Runnable 任務(wù)交給匿名內(nèi)部類的語(yǔ)法,此處是創(chuàng)建了一個(gè)類,實(shí)現(xiàn)Runnable,同時(shí)創(chuàng)建了實(shí)例,并且把實(shí)例傳給Thread的構(gòu)造方法(匿名內(nèi)部類的創(chuàng)建的實(shí)例作為Thread的構(gòu)造方法參數(shù))
4.4使用Lambda表達(dá)式
這是比較簡(jiǎn)單的,推薦的寫法
public class ThreadDemo5 { public static void main(String[] args) { Thread t = new Thread(() -> { System.out.println("hello world!!"); }); t.start(); } }
把任務(wù)通過(guò)Lambda表達(dá)式來(lái)描述,直接把Lambda表達(dá)式傳給Thread的構(gòu)造方法
到此這篇關(guān)于Java多線程Thread類的使用詳解的文章就介紹到這了,更多相關(guān)Java多線程Thread內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Spring?Boot?RestController接口輸出到終端的操作代碼
這篇文章主要介紹了Spring?Boot?RestController接口如何輸出到終端,使用?HttpServletResponse?類,可以在使用curl執(zhí)行?Spring?Boot?REST接口的同時(shí),在控制臺(tái)輸出一些信息,給運(yùn)維人員知道當(dāng)前命令執(zhí)行的狀態(tài),感興趣的朋友跟隨小編一起看看吧2023-09-09淺談java中BigDecimal的equals與compareTo的區(qū)別
下面小編就為大家?guī)?lái)一篇淺談java中BigDecimal的equals與compareTo的區(qū)別。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-11-11JDK-StringJoiner構(gòu)造及添加元素源碼分析
這篇文章主要為大家介紹了JDK-StringJoiner構(gòu)造及添加元素源碼分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-07-07java發(fā)送form-data請(qǐng)求實(shí)現(xiàn)文件上傳的示例代碼
最近做一個(gè)需求,需要請(qǐng)求第三方接口上傳文件,該請(qǐng)求類型是form-data請(qǐng)求,本文就來(lái)介紹一下java發(fā)送form-data請(qǐng)求實(shí)現(xiàn)文件上傳的示例代碼,感興趣的可以了解一下2023-12-12Java設(shè)計(jì)模式中的工廠及抽象工廠模式解析
這篇文章主要介紹了Java設(shè)計(jì)模式中的工廠及抽象工廠模式解析,工廠模式作為創(chuàng)建型設(shè)計(jì)模式中常見(jiàn)的設(shè)計(jì)方法,一般情況下,工廠模式分為3種,簡(jiǎn)單工作、工廠方法、抽象工作,其實(shí)簡(jiǎn)單工廠只是工廠方法的一種特例,需要的朋友可以參考下2023-12-12java實(shí)現(xiàn)后臺(tái)數(shù)據(jù)顯示在前端
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)后臺(tái)數(shù)據(jù)顯示在前端,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-02-02