(四)Thread.join的作用和原理

文章簡介

很多人對Thread.join的作用以及實現(xiàn)了解得很少粪躬,畢竟這個api我們很少使用毅否。這篇文章仍然會結(jié)合使用及原理進行深度分析

內(nèi)容導航

  1. Thread.join的作用
  2. Thread.join的實現(xiàn)原理
  3. 什么時候會使用Thread.join

Thread.join的作用

之前有人問過我一個這樣的面試題

Java中如何讓多線程按照自己指定的順序執(zhí)行?

這個問題最簡單的回答是通過Thread.join來實現(xiàn)秸脱,久而久之就讓很多人誤以為Thread.join是用來保證線程的順序性的簇秒。
下面這段代碼演示了Thread.join的作用

public class JoinDemo extends Thread{
    int i;
    Thread previousThread; //上一個線程
    public JoinDemo(Thread previousThread,int i){
        this.previousThread=previousThread;
        this.i=i;
    }
    @Override
    public void run() {
        try {
          //調(diào)用上一個線程的join方法瞬女,大家可以自己演示的時候可以把這行代碼注釋掉
            previousThread.join(); 
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("num:"+i);
    }
    public static void main(String[] args) {
        Thread previousThread=Thread.currentThread();
        for(int i=0;i<10;i++){
            JoinDemo joinDemo=new JoinDemo(previousThread,i);
            joinDemo.start();
            previousThread=joinDemo;
        }
    }
}

上面的代碼,注意 previousThread.join部分榕茧,大家可以把這行代碼注釋以后看看運行效果发乔,在沒有加join的時候運行的結(jié)果是不確定的。加了join以后雪猪,運行結(jié)果按照遞增的順序展示出來栏尚。

thread.join的含義是當前線程需要等待previousThread線程終止之后才從thread.join返回。簡單來說只恨,就是線程沒有執(zhí)行完之前译仗,會一直阻塞在join方法處。

下面的圖表現(xiàn)了join對于線程的作用

TIM圖片20181204180103.png

Thread.join的實現(xiàn)原理

線程是如何被阻塞的官觅?又是通過什么方法喚醒的呢纵菌?先來看看Thread.join方法做了什么事情

public class Thread implements Runnable {
    ...
    public final void join() throws InterruptedException {
        join(0);
    }
    ...
    public final synchronized void join(long millis) throws InterruptedException {
        long base = System.currentTimeMillis();
        long now = 0;
        if (millis < 0) {
            throw new IllegalArgumentException("timeout value is negative");
        }
        if (millis == 0) { //判斷是否攜帶阻塞的超時時間,等于0表示沒有設置超時時間
            while (isAlive()) {//isAlive獲取線程狀態(tài)休涤,無線等待直到previousThread線程結(jié)束
                wait(0); //調(diào)用Object中的wait方法實現(xiàn)線程的阻塞
            }
        } else { //阻塞直到超時
            while (isAlive()) { 
                long delay = millis - now;
                if (delay <= 0) {
                    break;
                }
                wait(delay);
                now = System.currentTimeMillis() - base;
            }
        }
    }
    ...

從join方法的源碼來看咱圆,join方法的本質(zhì)調(diào)用的是Object中的wait方法實現(xiàn)線程的阻塞笛辟,wait方法的實現(xiàn)原理我們在后續(xù)的文章再說詳細闡述。但是我們需要知道的是序苏,調(diào)用wait方法必須要獲取鎖手幢,所以join方法是被synchronized修飾的,synchronized修飾在方法層面相當于synchronized(this),this就是previousThread本身的實例忱详。

有很多人不理解join為什么阻塞的是主線程呢? 不理解的原因是阻塞主線程的方法是放在previousThread這個實例作用围来,讓大家誤以為應該阻塞previousThread線程。實際上主線程會持有previousThread這個對象的鎖匈睁,然后調(diào)用wait方法去阻塞监透,而這個方法的調(diào)用者是在主線程中的。所以造成主線程阻塞航唆。

第二個問題胀蛮,為什么previousThread線程執(zhí)行完畢就能夠喚醒住線程呢?或者說是在什么時候喚醒的糯钙?

要了解這個問題醇滥,我們又得翻jdk的源碼,但是如果大家對線程有一定的基本了解的話超营,通過wait方法阻塞的線程鸳玩,需要通過notify或者notifyall來喚醒。所以在線程執(zhí)行完畢以后會有一個喚醒的操作演闭,只是我們不需要關心不跟。
接下來在hotspot的源碼中找到 thread.cpp,看看線程退出以后有沒有做相關的事情來證明我們的猜想.

void JavaThread::exit(bool destroy_vm, ExitType exit_type) {
  assert(this == JavaThread::current(),  "thread consistency check");
  ...
  // Notify waiters on thread object. This has to be done after exit() is called
  // on the thread (if the thread is the last thread in a daemon ThreadGroup the
  // group should have the destroyed bit set before waiters are notified).
  ensure_join(this); 
  assert(!this->has_pending_exception(), "ensure_join should have cleared");
  ...

觀察一下 ensure_join(this)這行代碼上的注釋米碰,喚醒處于等待的線程對象窝革,這個是在線程終止之后做的清理工作,這個方法的定義代碼片段如下

static void ensure_join(JavaThread* thread) {
  // We do not need to grap the Threads_lock, since we are operating on ourself.
  Handle threadObj(thread, thread->threadObj());
  assert(threadObj.not_null(), "java thread object must exist");
  ObjectLocker lock(threadObj, thread);
  // Ignore pending exception (ThreadDeath), since we are exiting anyway
  thread->clear_pending_exception();
  // Thread is exiting. So set thread_status field in  java.lang.Thread class to TERMINATED.
  java_lang_Thread::set_thread_status(threadObj(), java_lang_Thread::TERMINATED);
  // Clear the native thread instance - this makes isAlive return false and allows the join()
  // to complete once we've done the notify_all below
  //這里是清除native線程吕座,這個操作會導致isAlive()方法返回false
  java_lang_Thread::set_thread(threadObj(), NULL);
  lock.notify_all(thread);//注意這里
  // Ignore pending exception (ThreadDeath), since we are exiting anyway
  thread->clear_pending_exception();
}

ensure_join方法中虐译,調(diào)用 lock.notify_all(thread); 喚醒所有等待thread鎖的線程,意味著調(diào)用了join方法被阻塞的主線程會被喚醒吴趴; 到目前為止漆诽,我們基本上對join的原理做了一個比較詳細的分析

總結(jié),Thread.join其實底層是通過wait/notifyall來實現(xiàn)線程的通信達到線程阻塞的目的锣枝;當線程執(zhí)行結(jié)束以后厢拭,會觸發(fā)兩個事情,第一個是設置native線程對象為null撇叁、第二個是通過notifyall方法供鸠,讓等待在previousThread對象鎖上的wait方法被喚醒。

什么時候會使用Thread.join

在實際應用開發(fā)中陨闹,我們很少會使用thread.join楞捂。在實際使用過程中薄坏,我們可以通過join方法來等待線程執(zhí)行的結(jié)果,其實有點類似future/callable的功能寨闹。
我們通過以下偽代碼來說明join的使用場景

public void joinDemo(){
   //....
   Thread t=new Thread(payService);
   t.start();
   //.... 
   //其他業(yè)務邏輯處理,不需要確定t線程是否執(zhí)行完
   insertData();
   //后續(xù)的處理胶坠,需要依賴t線程的執(zhí)行結(jié)果,可以在這里調(diào)用join方法等待t線程執(zhí)行結(jié)束
   t.join();
}

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末鼻忠,一起剝皮案震驚了整個濱河市涵但,隨后出現(xiàn)的幾起案子杈绸,更是在濱河造成了極大的恐慌帖蔓,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,858評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件瞳脓,死亡現(xiàn)場離奇詭異塑娇,居然都是意外死亡,警方通過查閱死者的電腦和手機劫侧,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,372評論 3 395
  • 文/潘曉璐 我一進店門埋酬,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人烧栋,你說我怎么就攤上這事写妥。” “怎么了审姓?”我有些...
    開封第一講書人閱讀 165,282評論 0 356
  • 文/不壞的土叔 我叫張陵珍特,是天一觀的道長。 經(jīng)常有香客問我魔吐,道長扎筒,這世上最難降的妖魔是什么酬姆? 我笑而不...
    開封第一講書人閱讀 58,842評論 1 295
  • 正文 為了忘掉前任嗜桌,我火速辦了婚禮辞色,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘相满。我一直安慰自己诱篷,他們只是感情好,可當我...
    茶點故事閱讀 67,857評論 6 392
  • 文/花漫 我一把揭開白布雳灵。 她就那樣靜靜地躺著,像睡著了一般悯辙。 火紅的嫁衣襯著肌膚如雪迎吵。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,679評論 1 305
  • 那天针贬,我揣著相機與錄音击费,去河邊找鬼。 笑死桦他,一個胖子當著我的面吹牛蔫巩,可吹牛的內(nèi)容都是我干的快压。 我是一名探鬼主播,決...
    沈念sama閱讀 40,406評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼坪郭,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了歪沃?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,311評論 0 276
  • 序言:老撾萬榮一對情侶失蹤沪曙,失蹤者是張志新(化名)和其女友劉穎萎羔,沒想到半個月后液走,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體外驱,經(jīng)...
    沈念sama閱讀 45,767評論 1 315
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,945評論 3 336
  • 正文 我和宋清朗相戀三年磅崭,在試婚紗的時候發(fā)現(xiàn)自己被綠了瓦哎。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片砸喻。...
    茶點故事閱讀 40,090評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡蒋譬,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出犯助,到底是詐尸還是另有隱情,我是刑警寧澤剂买,帶...
    沈念sama閱讀 35,785評論 5 346
  • 正文 年R本政府宣布癌蓖,位于F島的核電站婚肆,受9級特大地震影響租副,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜较性,卻給世界環(huán)境...
    茶點故事閱讀 41,420評論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望赞咙。 院中可真熱鬧,春花似錦人弓、人聲如沸着逐。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,988評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至秀姐,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間省有,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,101評論 1 271
  • 我被黑心中介騙來泰國打工伸头, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人恤磷。 一個月前我還...
    沈念sama閱讀 48,298評論 3 372
  • 正文 我出身青樓野宜,卻偏偏與公主長得像,于是被迫代替她去往敵國和親匈子。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 45,033評論 2 355

推薦閱讀更多精彩內(nèi)容

  • 本文是我自己在秋招復習時的讀書筆記游岳,整理的知識點搁吓,也是為了防止忘記吭历,尊重勞動成果,轉(zhuǎn)載注明出處哦晌区!如果你也喜歡,那...
    波波波先森閱讀 11,268評論 4 56
  • 進程和線程 進程 所有運行中的任務通常對應一個進程,當一個程序進入內(nèi)存運行時,即變成一個進程.進程是處于運行過程中...
    勝浩_ae28閱讀 5,110評論 0 23
  • 一恼五、進程和線程 進程 進程就是一個執(zhí)行中的程序?qū)嵗扌福總€進程都有自己獨立的一塊內(nèi)存空間灾馒,一個進程中可以有多個線程遣总。...
    阿敏其人閱讀 2,612評論 0 13
  • 本文主要講了java中多線程的使用方法、線程同步旭斥、線程數(shù)據(jù)傳遞、線程狀態(tài)及相應的一些線程函數(shù)用法垂券、概述等。在這之前...
    4ea0af17fd67閱讀 589評論 2 17
  • 2017.6.24 星期六 晴 淡定 今天把A芳老師的點贊有了一條算芯,突然就好多人自動來加我,還是錢吸引力大凳宙,幾分鐘...
    愛林FAB閱讀 175評論 0 0