17.Thread.join的用法和原理

Posted 纵横千里,捭阖四方

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了17.Thread.join的用法和原理相关的知识,希望对你有一定的参考价值。

在应用程序中,如果某段程序希望等待前面的线程执行结束后再执行,并发编程里有很多工具可以做,其中Join()就可以 。 join的作用就是让线程的执行结果对后续线程的访问可见。看个例子:

public class JoinTest 
    public static int i = 10;

    public static void main(String[] args) throws InterruptedException 
        Thread thread = new Thread(() -> 
            i = 30;
            System.out.println("sub thread");
        );
        thread.start();
//        thread.join();
        System.out.println("main " + i);

    

在上面的代码中,如果我们注释掉thread.join(),打印的结果是:

main 10
sub thread

如果取消注释,输出的结果是:

sub thread
main 30

为什么会这样呢?

这就是因为正常情况下,子线程还没执行,main线程已经完了,而加了thread.Join就会让主线程等待,直到子线程的任务都完成之后再继续进行Join()之后的内容。画成结构图就是:

1 经典面试题:多线程按顺序执行

在多线程中有多种方法让线程按特定顺序执行,你可以用线程类的join()方法在一个线程中启动另一个 线程,另外一个线程完成该线程继续执行。为了确保三个线程的顺序你应该先启动最后一个(T3调用 T2,T2调用T1),这样T1就会先完成而T3最后完成。 实际上先启动三个线程中哪一个都行, 因为在每个线程的run方法中用join方法限定了三个线程的执行顺序。

public class JoinTest2 
    public static void main(String[] args) 
        final Thread t1 = new Thread(new Runnable() 
            public void run() 
                System.out.println("t1");
            
        );

        final Thread t2 = new Thread(new Runnable() 
            public void run() 
                try 
                    t1.join();
                 catch (InterruptedException e) 
                    e.printStackTrace();
                
                System.out.println("t2");
            
        );

        final Thread t3 = new Thread(new Runnable() 
            public void run() 
                try 
                    t2.join();
                 catch (InterruptedException e) 
                    e.printStackTrace();
                
                System.out.println("t3");
            
        );

        t1.start();
        t2.start();
        t3.start();
    

2 执行原理

从上面的分析可以发现,Thread.Join的本质就是通过阻塞唤醒的方式来实现的,底层可能就是使用wait()/notify()方法,实际也确实如此。Join()的核心代码如下:

 

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) 
        while (isAlive()) //如果线程是存活状态
            wait(0);//就调用wait()方法阻塞当前线程
        
     else 
        while (isAlive()) 
            long delay = millis - now;
            if (delay <= 0) 
                break;
            
            wait(delay);
            now = System.currentTimeMillis() - base;
        
    

既然存在wait()方法阻塞,那么必然存在notify()/notifyAll()方法唤醒,而基于Join()方法的原理,就是在线程终止之后触发这个动作。当然可以代码我们看不到,因为这是在JVM里实现的。我们就不再细看了。

以上是关于17.Thread.join的用法和原理的主要内容,如果未能解决你的问题,请参考以下文章

SSH原理与运用

SSH原理与运用:远程登录

转:WPF中ListBox的创建和多种绑定用法

SSH远程登录原理与运用

数组的多种用法

ansible playbook中when的多种用法和playbook handler