一、概述

join()是Thread类中的一个方法,它的作用是将当前线程挂起,等待其他线程结束后再执行当前线程,即当前线程等待另一个调用join()方法的线程执行结束后再往下执行。通常用于在main主线程内,等待其它调用join()方法的线程执行结束再继续执行main主线程。本文将探索join方法的使用方式和使用原理。

二、join方法使用

2.1 join()示意图

JAVA多个线程 join java多线程join方法_System


上述案例示意图中,主线程A入栈运行,运行时创建子线程B和线程C执行任务,B线程或C线程调用join方法,阻塞当前线程A,B或C线程执行完成,A线程才会继续执行。

2.2 案例代码

public class ThreadB extends Thread {

    public void run() {
        System.out.println("线程:" + Thread.currentThread().getName() + "休眠:10s");
        try {
            TimeUnit.SECONDS.sleep(10);
        } catch (Exception e) {
            e.printStackTrace();
        }
        System.out.println("线程:" + Thread.currentThread().getName() + "休眠结束");
    }

}

public class ThreadC extends Thread {

    public void run() {
        System.out.println("线程:" + Thread.currentThread().getName() + "休眠:1s");
        try {
            TimeUnit.SECONDS.sleep(1);
        } catch (Exception e) {
            e.printStackTrace();
        }
        System.out.println("线程:" + Thread.currentThread().getName() + "休眠结束");
    }

}

public class Test {
    public static void main(String[] args) throws InterruptedException {
        System.out.println("主线程线程A开始运行。。。");
        ThreadB threadB = new ThreadB();
        threadB.setName("ThreadB");
        threadB.start();

        ThreadC threadC = new ThreadC();
        threadC.setName("ThreadC");
        threadC.start();

        threadB.join();

        System.out.println("主线程线程A结束运行");

    }
}

运行结果如下:

JAVA多个线程 join java多线程join方法_开发语言_02


方法join的作用是使所属的线程对象x正常执行run()方法中的任务,而使当前线程z进行无限期的阻塞,等待线程x被销毁后再继续执行线程z后面的代码。

2.3 join方法原理

先看一下join()方法的源码,源码基于JDK 1.8,具体如下:

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) {
           //判断线程是否存活,若为true,调用wait()方法阻塞
            while (isAlive()) {
                wait(0);
            }
        } else {
            //阻塞指定时间
            while (isAlive()) {
                long delay = millis - now;
                if (delay <= 0) {
                    break;
                }
                wait(delay);
                now = System.currentTimeMillis() - base;
            }
        }
    }

由上述源码可知,join方法的底层实际上还是调用了wait()方法,通过wait()方法实现线程阻塞等待,那么这里有个问题,阻塞完成后,如何通知呢?我们知道调用wait阻塞的线程,需要notify()或者notifyAll来进行唤醒,这里并没有显式调用这两个方法,那是如何实现的呢?这里涉及一个知识:

在java中,Thread类线程执行完run()方法后,会自动执行notifyAll()方法。