欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

一文詳解Java線程的6種狀態(tài)與生命周期

 更新時(shí)間:2022年05月10日 16:32:56   作者:淵渟岳  
一個(gè)線程在給定的時(shí)間點(diǎn)只能處于一種狀態(tài)。線程可以有6種狀態(tài):New、Runnable、Blocked、Waiting、Timed?waiting和Terminated。本文將詳細(xì)講解這6種狀態(tài),需要的可以參考一下

1.線程狀態(tài)(生命周期)

一個(gè)線程在給定的時(shí)間點(diǎn)只能處于一種狀態(tài)。

線程可以有如下6 種狀態(tài):

  • New (新創(chuàng)建):未啟動(dòng)的線程;
  • Runnable (可運(yùn)行):可運(yùn)行的線程,需要等待操作系統(tǒng)資源;
  • Blocked (被阻塞):等待監(jiān)視器鎖而被阻塞的線程;
  • Waiting (等待):等待喚醒狀態(tài),無限期地等待另一個(gè)線程喚醒;
  • Timed waiting (計(jì)時(shí)等待):在指定的等待時(shí)間內(nèi)等待另一個(gè)線程執(zhí)行操作的線程;
  • Terminated (被終止):已退出的線程。

要確定一個(gè)線程的當(dāng)前狀態(tài), 可調(diào)用getState 方法

線程狀態(tài)關(guān)系圖

注意:虛線框(全大寫英文)的狀態(tài)為Java線程狀態(tài)。

圖片

2.操作線程狀態(tài)

2.1.新創(chuàng)建狀態(tài)(NEW)

就是實(shí)例化線程完成后,未啟動(dòng)線程的狀態(tài)。

可通過三種方式創(chuàng)建線程

  • 重寫Thread類run()方法
  • 實(shí)現(xiàn)Runnable接口
  • 實(shí)現(xiàn)Callable接口

一個(gè)簡單的例子概括三種方式

public?class?Demo?{

????public?static?void?main(String[]?args)?throws?ExecutionException,?InterruptedException?{
????????/**
?????????*?1.直接重寫run()?或繼承Thread類再重寫run()
?????????*/
????????Thread?thread?=?new?Thread()?{
????????????@Override
????????????public?void?run()?{
????????????????System.out.println("Thread");
????????????}
????????};
????????//?開啟線程
????????thread.start();

????????/**
?????????*?2.lambda、內(nèi)部類或線程類方式實(shí)現(xiàn)Runnable接口,實(shí)現(xiàn)run()方法
?????????*?再交給Thread?類
?????????*/
????????Thread?runThread?=?new?Thread(()?->?{
????????????System.out.println("Runnable");
????????});
????????//?開啟線程
????????runThread.start();

????????/**
?????????*?3.lambda、內(nèi)部類或線程類方式實(shí)現(xiàn)Callable接口,實(shí)現(xiàn)call()方法
?????????*?再交給Thread 類:FutureTask本質(zhì)也是Runnable實(shí)現(xiàn)類
?????????*/
????????FutureTask<String>?futureTask?=?new?FutureTask<String>(()?->?{
????????????System.out.println("Callable");
????????????return?"CallableThread";
????????});
????????Thread?callThread?=?new?Thread(futureTask);
????????//?開啟線程
????????callThread.start();
????????//?獲取call()方法的返回值
????????String?s?=?futureTask.get();
????????System.out.println("call()方法的返回值:"+s);
????}

}

不重寫 run() 或 call() 方法直接實(shí)例化Thread類創(chuàng)建的線程沒有實(shí)際意義;

只有Callable方式創(chuàng)建的線程可以獲取線程的返回值。

2.2.可運(yùn)行狀態(tài)(RUNNABLE)

該狀態(tài)指的是線程實(shí)例化對(duì)象調(diào)用start()方法后進(jìn)入的狀態(tài)。線程處于可以運(yùn)行狀態(tài),如果有處理器等資源,就可以執(zhí)行程序。

該狀態(tài)在操作系統(tǒng)層面包含兩步:線程就緒和線程運(yùn)行中,但在Java線程狀態(tài)中,這兩步都統(tǒng)稱為Runnable(可運(yùn)行)狀態(tài)。

線程由就緒狀態(tài)變?yōu)檫\(yùn)行狀態(tài),重點(diǎn)就看你的線程有沒有搶到CPU資源(CPU時(shí)間片),誰搶到就運(yùn)行,沒搶到就等。因?yàn)镃PU時(shí)間片(執(zhí)行時(shí)間)非常短,大概十幾毫秒,所以線程切換的這個(gè)時(shí)間是非常短的,就緒狀態(tài)變?yōu)檫\(yùn)行狀態(tài)的時(shí)間也非常短,在開發(fā)時(shí)幾乎感覺不到這種狀態(tài)的變化,所以在Java中將兩者看作是一個(gè)整體,重點(diǎn)關(guān)注線程可否運(yùn)行并區(qū)別于其他狀態(tài)即可,更進(jìn)一步簡化線程的開發(fā)。如果你的程序要運(yùn)行很久(比如寫個(gè)死循環(huán)),在一個(gè)CPU時(shí)間片內(nèi)沒有執(zhí)行完成,那么你的線程就要搶下一次的CPU時(shí)間片,搶到了才可以繼續(xù)執(zhí)行程序,沒搶到那就要繼續(xù)搶,直到線程中的程序執(zhí)行完成。

其實(shí)這個(gè)場(chǎng)景應(yīng)該都見到過,例如多個(gè)線程執(zhí)行同一個(gè)程序,都將日志打印到同一個(gè)文件時(shí),就會(huì)出現(xiàn)不同線程的日志混在了一起的情況,不利于排查問題。解決這種問題常見的方法有:一是分線程打印日志到不同文件;二是將日志信息保存到字符串對(duì)象中,在程序的最后將日志信息一次性打印到文件。第二種方式就是利用CPU的一個(gè)時(shí)間片來完成日志信息的打印。

注意:程序只能對(duì)新建狀態(tài)的線程調(diào)用start()方法,不要對(duì)處于非新建狀態(tài)的線程調(diào)用start() 方法,這都會(huì)引發(fā)IllegalThreadStateException異常。

2.3.被阻塞狀態(tài)(BLOCKED)

線程處于等待監(jiān)視器而被阻塞的狀態(tài)。有一個(gè)線程獲取了鎖未釋放,其他線程也來獲取,但發(fā)現(xiàn)獲取不到鎖也進(jìn)入了被阻塞狀態(tài)。

被阻塞狀態(tài)只存在于多線程并發(fā)訪問下,區(qū)別于后面兩種因線程自己進(jìn)入”等待“而導(dǎo)致的阻塞。

進(jìn)入狀態(tài)

  • 進(jìn)入synchronized 代碼塊/方法
  • 未獲取到鎖

退出狀態(tài)

  • 獲取到監(jiān)視器鎖

2.4.等待喚醒狀態(tài)(WAITING)

整個(gè)流程是這樣的:線程在某個(gè)對(duì)象的同步方法中先獲取到對(duì)象鎖;在執(zhí)行wait方法時(shí),該線程將釋放對(duì)象鎖,并且該線程被放入到這個(gè)對(duì)象的等待隊(duì)列;等待另一個(gè)線程獲取到同一個(gè)對(duì)象的鎖,然后通過notify() 或 notifyAll() 方法喚醒對(duì)象等待隊(duì)列中的線程。

從整個(gè)流程可以知道

wait (),notify () 和 notifyAll () 方法需要在線程獲取到鎖的情況下才可以繼續(xù)執(zhí)行,所以這三個(gè)方法都需要放在同步代碼塊/方法中執(zhí)行,否則報(bào)異常:java.lang.IllegalMonitorStateException。

在同步代碼塊中,線程進(jìn)入WAITING 狀態(tài)時(shí),鎖會(huì)被釋放,不會(huì)導(dǎo)致該線程阻塞。反過來想下,如果鎖沒釋放,那其他線程就沒辦法獲取鎖,也就沒辦法喚醒它。

進(jìn)入狀態(tài)

  • object.wait()
  • thread.join()
  • LockSupport.park()

退出狀態(tài)

  • object.notify()
  • object.notifyall()
  • LockSupport.unpark()

2.5.計(jì)時(shí)等待狀態(tài)(TIMED_WAITING)

一般是計(jì)時(shí)結(jié)束就會(huì)自動(dòng)喚醒線程繼續(xù)執(zhí)行后面的程序,對(duì)于Object.wait(long) 方法還可以主動(dòng)通知喚醒。

注意:Thread類下的sleep() 方法可以放在任意地方執(zhí)行;而wait(long) 方法和wait() 方法一樣,需要放在同步代碼塊/方法中執(zhí)行,否則報(bào)異常:java.lang.IllegalMonitorStateException。

進(jìn)入狀態(tài)

  • Thread.sleep(long)
  • Object.wait(long)
  • Thread.join(long)
  • LockSupport.parkNanos(long)
  • LockSupport.parkNanos(Object blocker, long nanos)
  • LockSupport.parkUntil(long)
  • LockSupport.parkUntil(Object blocker, long deadline)

注:blocker 參數(shù)為負(fù)責(zé)此線程駐留的同步對(duì)象。

退出狀態(tài)

  • 計(jì)時(shí)結(jié)束
  • LockSupport.unpark(Thread)
  • object.notify()
  • object.notifyall()

2.6.終止(TERMINATED)

線程執(zhí)行結(jié)束

  • run()/call() 執(zhí)行完成
  • stop()線程
  • 錯(cuò)誤或異常>>意外死亡

stop() 方法已棄用。

3.查看線程的6種狀態(tài)

通過一個(gè)簡單的例子來查看線程出現(xiàn)的6種狀態(tài)。

案例

public?class?Demo3?{
????private?static?Object?object?="obj";
????
????public?static?void?main(String[]?args)?throws?InterruptedException?{

????????Thread?thread0?=?new?Thread(()?->?{
????????????try?{
????????????????//?被阻塞狀態(tài)(BLOCKED)
????????????????synchronized?(object){
????????????????????System.out.println("thread0?進(jìn)入:等待喚醒狀態(tài)(WAITING)");
????????????????????object.wait();
????????????????????System.out.println("thread0?被解除完成:等待喚醒狀態(tài)(WAITING)");
????????????????}
????????????????System.out.println("thread0?"+Thread.currentThread().getState());
????????????}?catch?(InterruptedException?e)?{
????????????????e.printStackTrace();
????????????}
????????});
????????//?新創(chuàng)建狀態(tài)(NEW)
????????System.out.println(thread0.getName()+":"+thread0.getState());

????????Thread?thread1?=?new?Thread(()?->?{
????????????try?{
????????????????System.out.println("thread1 進(jìn)入:計(jì)時(shí)等待狀態(tài)(TIMED_WAITING)");
????????????????Thread.sleep(2);
????????????????System.out.println("thread1 出來:計(jì)時(shí)等待狀態(tài)(TIMED_WAITING)");
????????????}?catch?(InterruptedException?e)?{
????????????????e.printStackTrace();
????????????}
????????????//?被阻塞狀態(tài)(BLOCKED)
????????????synchronized?(object){
????????????????System.out.println("thread1 解除:等待喚醒狀態(tài)(WAITING)");
????????????????object.notify();
????????????????System.out.println("thread1 解除完成:等待喚醒狀態(tài)(WAITING)");
????????????}
????????????System.out.println("thread1?"+Thread.currentThread().getState());
????????});
????????//?新創(chuàng)建狀態(tài)(NEW)
????????System.out.println(thread1.getName()+":"+thread1.getState());

????????printState(thread0);
????????printState(thread1);

????????//?可運(yùn)行狀態(tài)(RUNNABLE)
????????thread0.start();
????????//?可運(yùn)行狀態(tài)(RUNNABLE)
????????thread1.start();

????}
????
????
????//?使用獨(dú)立線程來打印線程狀態(tài)
????private?static?void?printState(Thread?thread)?{
????????new?Thread(()->{
????????????while?(true){
????????????????System.out.println(thread.getName()+":"+thread.getState());
????????????????if?(thread.getState().equals(Thread.State.TERMINATED)){
????????????????????System.out.println(thread.getName()+":"+thread.getState());
????????????????????break;
????????????????}
????????????}
????????}).start();
????}
}

執(zhí)行結(jié)果:簡化后的輸出結(jié)果

Thread-0:NEW
Thread-1:NEW
Thread-0:RUNNABLE
Thread-1:RUNNABLE
thread0 進(jìn)入:等待喚醒狀態(tài)(WAITING)
Thread-1:BLOCKED
thread1 進(jìn)入:計(jì)時(shí)等待狀態(tài)(TIMED_WAITING)
Thread-0:BLOCKED
Thread-0:WAITING
……
Thread-0:WAITING
Thread-1:BLOCKED
Thread-1:TIMED_WAITING
……
Thread-1:TIMED_WAITING
Thread-1:BLOCKED
……
Thread-1:BLOCKED
Thread-0:WAITING
……
Thread-0:WAITING
thread1 出來:計(jì)時(shí)等待狀態(tài)(TIMED_WAITING)
Thread-0:WAITING
Thread-1:BLOCKED
thread1 解除:等待喚醒狀態(tài)(WAITING)
Thread-1:BLOCKED
Thread-0:WAITING
Thread-0:BLOCKED
thread1 解除完成:等待喚醒狀態(tài)(WAITING)
Thread-1:BLOCKED
thread1 RUNNABLE
Thread-0:BLOCKED
Thread-1:TERMINATED
thread0 被解除完成:等待喚醒狀態(tài)(WAITING)
Thread-0:BLOCKED
thread0 RUNNABLE
Thread-0:TERMINATED

圖片

最終的執(zhí)行結(jié)果如圖。

注意:因?yàn)榘咐惺褂昧霜?dú)立線程來打印不同線程的狀態(tài),會(huì)出現(xiàn)狀態(tài)打印稍微延遲的情況。

到此這篇關(guān)于一文詳解Java線程的6種狀態(tài)與生命周期的文章就介紹到這了,更多相關(guān)Java線程狀態(tài)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 使用Maven Helper解決Maven插件沖突的方法

    使用Maven Helper解決Maven插件沖突的方法

    這篇文章主要介紹了使用Maven Helper解決Maven插件沖突的方法,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2018-12-12
  • Java實(shí)現(xiàn)的斷點(diǎn)續(xù)傳功能的示例代碼

    Java實(shí)現(xiàn)的斷點(diǎn)續(xù)傳功能的示例代碼

    本篇文章主要介紹了Java實(shí)現(xiàn)的斷點(diǎn)續(xù)傳功能的示例代碼,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2017-02-02
  • Java如何實(shí)現(xiàn)支付寶電腦支付基于servlet版本

    Java如何實(shí)現(xiàn)支付寶電腦支付基于servlet版本

    這篇文章主要介紹了Java如何實(shí)現(xiàn)支付寶電腦支付基于servlet版本,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2019-11-11
  • 詳解Java中Callable和Future的區(qū)別

    詳解Java中Callable和Future的區(qū)別

    這篇文章主要介紹了Java中Callable和Future的區(qū)別,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2022-11-11
  • java高并發(fā)之線程的基本操作詳解

    java高并發(fā)之線程的基本操作詳解

    本文主要介紹java高并發(fā)線程的基本操作,這里整理詳細(xì)的資料來解釋線程的知識(shí),有需要的學(xué)習(xí)高并發(fā)的朋友可以參考下
    2021-10-10
  • JAVA各種加密與解密方式總結(jié)大全

    JAVA各種加密與解密方式總結(jié)大全

    這篇文章主要給大家介紹了關(guān)于JAVA各種加密與解密方式總結(jié)的相關(guān)資料,加密是指對(duì)原來為明文的文件或數(shù)據(jù)按某種算法進(jìn)行處理,使其成為不可讀的一段代碼,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2023-07-07
  • SpringBoot實(shí)現(xiàn)日志鏈路追蹤的項(xiàng)目實(shí)踐

    SpringBoot實(shí)現(xiàn)日志鏈路追蹤的項(xiàng)目實(shí)踐

    在分布式系統(tǒng)中,由于請(qǐng)求的處理過程可能會(huì)跨越多個(gè)服務(wù),因此,對(duì)請(qǐng)求的追蹤變得尤為重要,本文主要介紹了SpringBoot實(shí)現(xiàn)日志鏈路追蹤的項(xiàng)目實(shí)踐,感興趣的可以了解一下
    2024-03-03
  • Java 正則表達(dá)式詳細(xì)介紹

    Java 正則表達(dá)式詳細(xì)介紹

    本文主要介紹 Java 正則表達(dá)式的內(nèi)容,這里整理了Java 正則表達(dá)式的相關(guān)資料,并詳細(xì)介紹,附有代碼示例,有興趣的小伙伴可以參考下
    2016-09-09
  • java之項(xiàng)目運(yùn)維手冊(cè)

    java之項(xiàng)目運(yùn)維手冊(cè)

    這篇文章主要介紹了java之項(xiàng)目運(yùn)維手冊(cè),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-04-04
  • java多線程編程必備volatile與synchronized深入理解

    java多線程編程必備volatile與synchronized深入理解

    這篇文章主要介紹了java多線程編程必備volatile與synchronized的深入理解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-04-04

最新評(píng)論