join方法原理

导读:本篇文章讲解 join方法原理,希望对大家有帮助,欢迎收藏,转发!站点地址:www.bmabk.com

join()方法–原理同wait方法

如果不知道保护性暂停是啥的可以参考一下上一篇文章
https://www.cnblogs.com/duizhangz/p/16222854.html

join方法本质上和加了超时的保护性暂停差不多。

首先抛出join方法使用场景即保证线程的顺序执行。

public static void main(String[] args) throws InterruptedException {
	Thread t1 = new Thread(() -> {
        try {
            Thread.sleep(2000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    },"t1");
    t1.join();
    log.debug("t1 join结束");
    log.debug("继续执行主线程方法");
}

首先我们调用的是t1对象的join()方法,我们首先会知道这个main的主线程会等待t1线程执行结束后才会接着执行。

下面直接抛出源码join()方法,如果join未加参数就是执行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) {
        while (isAlive()) {   // isAlive 就是判断调用该方法的线程是否存活
            wait(0);
        }
    } else {
        while (isAlive()) {
            long delay = millis - now;
            if (delay <= 0) {
                break;
            }
            wait(delay);
            now = System.currentTimeMillis() - base;
        }
    }
}

主要是两个条件判断,一个是当millis参数为0时,就会进入一个wait(0),不为0时就会和上面超时保护差不多的代码中。

首先我们是在main线程中,调用了t1线程对象的join()方法,所以我们需要搞清楚两个状态。

  1. 当前join()方法是synchronized修饰的,所以当前的wait方法锁住的锁资源是t1对象。
  2. 然后main主线程就会因为wait()方法进入waitting状态。直到有人调用了t1.notify方法唤醒了main线程。

这个逻辑比较搞,就这线程那线程的,刚开始我都被搞乱了,仔细想想就OK了。

刚开始学的时候,我就觉得奇怪,t1线程结束是怎么唤醒main线程呢?现在理清楚了就是需要锁资源t1对象调用notifyAll()方法,然而在JVM底层当t1线程运行结束时就会调用该线程对象的notifyAll()方法,将被t1.wait()方法锁住的main线程。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/17352.html

(0)
小半的头像小半

相关推荐

极客之音——专业性很强的中文编程技术网站,欢迎收藏到浏览器,订阅我们!