Thread.join()

本文深入探讨了Java中Thread类的join()方法的使用场景、作用及其实现原理。join()方法使主线程等待子线程执行完毕,避免主线程提前结束导致的数据不一致问题。文章通过具体代码示例展示了如何正确使用join()方法,并解释了其三个重载版本的区别。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

 

为什么要用join()方法

如果子线程里要进行大量的耗时的运算,主线程往往将于子线程之前结束,但是如果主线程处理完其他的事务后,需要用到子线程的处理结果,也就是 主线程需要等待子线程执行完成之后再结束,这个时候就要用到join()方法了

join() 的作用

让父线程等待子线程结束之后才能继续运行

join() 一共有三个重载版本,分别是无参、一个参数、两个参数:

1 public final void join() throws InterruptedException;

2 public final synchronized void join(long millis) throws InterruptedException;

3 public final synchronized void join(long millis, int nanos) throws InterruptedException;
实列: 

public class TestJoin {

    public static void main(String[] args) throws InterruptedException {
        // TODO Auto-generated method stub
        ThreadTest t1=new ThreadTest("A");
        ThreadTest t2=new ThreadTest("B");
        t1.start();
        t2.start();
    }


}
class ThreadTest extends Thread {
    private String name;

    public ThreadTest(String name) {
        this.name = name;
    }

    public void run() {
        for (int i = 1; i <= 5; i++) {
            System.out.println(name + "-" + i);
        }
    }
}

结果: 可能无 t1 线程没有执行完毕 就执行t2

 

加上 join() 在主线程中调用t.Join(),也就是在主线程中加入了t线程的代码,必须让t线程执行完毕之后,主线程(调用方)才能正常执行

public class JoinTest {
    public static void main(String[] arg) {
        ThreadTests t = new ThreadTests("Li");
        ThreadTests t2=new ThreadTests("cheng");
        t.start();
        System.out.println("t");
        try {
            t.join();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

        System.out.println("t2");
        t2.start();




    }


}

class ThreadTests extends Thread {
    private String name;
    public ThreadTests(String name){
        this.name=name;
    }

    public void run() {
        for (int i = 1; i <= 2; i++) {
            System.out.println("name:"+name+"i="+ i);
        }
    }
}
结果:

t
name:Lii=1
name:Lii=2
t2
name:chengi=1
name:chengi=2

其中

(1) 三个方法都被final修饰,无法被子类重写。

(2) join(long), join(long, long) 是synchronized method,同步的对象是当前线程实例。

(2) 无参版本和两个参数版本最终都调用了一个参数的版本。

(3) join() 和 join(0) 是等价的,表示一直等下去;join(非0)表示等待一段时间。

从源码可以看到 join(0) 调用了Object.wait(0),其中Object.wait(0) 会一直等待,直到被notify/中断才返回。

while(isAlive())是为了防止子线程伪唤醒(spurious wakeup),只要子线程没有TERMINATED的,父线程就需要继续等下去。

(4) join() 和 sleep() 一样,可以被中断(被中断时,会抛出 InterrupptedException 异常);不同的是,join() 内部调用了 wait(),会出让锁,而 sleep() 会一直保持锁。

 



 

### Java 中 `Thread.join` 方法的使用 在多线程编程中,有时主线程需要等待其他子线程完成后再继续执行。此时可以使用 `join()` 方法来实现这一功能。当调用某个线程对象的 `join()` 方法时,当前线程会阻塞直到该线程结束[^1]。 #### 基本语法 ```java public final void join() throws InterruptedException; ``` 此方法会使当前正在运行的线程暂停并等待指定的线程终止。如果抛出了 `InterruptedException`,则表示另一个线程已中断了当前线程。 #### 使用示例 下面是一个简单的例子展示如何利用 `join()` 来确保两个并发工作的线程按顺序完成: ```java class MyThread extends Thread { private String name; public MyThread(String name) { this.name = name; } @Override public void run() { System.out.println(name + " is running"); try { // Simulate some work being done by sleeping. sleep(2000); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(name + " has finished its task."); } } public class JoinExample { public static void main(String[] args) { MyThread t1 = new MyThread("Thread-2"); t1.start(); // Start first thread try { t1.join(); // Wait until t1 finishes before starting next line } catch (InterruptedException e) { e.printStackTrace(); } t2.start(); // Only starts after t1 completes try { t2.join(); // Wait again for second thread to finish } catch (InterruptedException e) { e.printStackTrace(); } System.out.println("Both threads have completed their tasks."); } } ``` 在这个程序里,通过两次调用了 `t1.join()` 和 `t2.join()` ,保证了只有在线程 T1 完成之后才会启动T2,并且整个应用程序会在所有的工作都完成后才打印最终的消息。 #### 解决线程同步问题的方法 除了上述提到的方式外,在处理更复杂的场景下还可以考虑采用更高层次抽象工具如 `CountDownLatch`, `CyclicBarrier` 或者是基于锁机制(`ReentrantLock`)来进行更加精细控制下的协调操作[^4]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值