转自:https://www.jianshu.com/p/fc51be7e5bc0

原文解释的有点歧义,注意看原文评论。本文在原文基础上修改部分语句。

文章简介

很多人对Thread.join的作用以及实现了解得很少,毕竟这个api我们很少使用。这篇文章仍然会结合使用及原理进行深度分析

内容导航

  1. Thread.join的作用
  2. Thread.join的实现原理
  3. 什么时候会使用Thread.join

Thread.join的作用

之前有人问过我一个这样的面试题

Java中如何让多线程按照自己指定的顺序执行?

这个问题最简单的回答是通过Thread.join来实现,久而久之就让很多人误以为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;}@Overridepublic void run() {try {//调用上一个线程的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的时候运行的结果是不确定的。加了join以后,运行结果按照递增的顺序展示出来。

thread.join的含义是当前线程需要等待previousThread线程终止之后才从thread.join返回。简单来说,就是线程没有执行完之前,会一直阻塞在join方法处。

下面的图表现了join对于线程的作用

TIM图片20181204180103.png

Thread.join的实现原理

线程是如何被阻塞的?又是通过什么方法唤醒的呢?先来看看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获取线程状态,无限等待直到previousThread线程结束wait(0); //调用Object中的wait方法实现线程的阻塞}} else { //阻塞直到超时while (isAlive()) { long delay = millis - now;if (delay <= 0) {break;}wait(delay);now = System.currentTimeMillis() - base;}}}...

从join方法的源码来看,join方法的本质调用的是Object中的wait方法实现线程的阻塞,wait方法的实现原理我们在后续的文章再说详细阐述。但是我们需要知道的是,调用wait方法必须要获取锁,所以join方法是被synchronized修饰的,synchronized修饰在方法层面相当于synchronized(this),this就是previousThread本身的实例。

当线程执行到Thread N时,相对它来说previousThread = Thread N-1。在执行Thread N线程内部方法时,就会处于阻塞状态,即因为previousThread.join()这一行代码导致的。当previousThread即Thread N-1线程执行完毕后,接着就执行Thread N中previousThread.join()之后的内容了。

实际上Thread N会持有previousThread这个对象的锁,然后调用wait方法去阻塞,而这个方法的调用者是在Thread N中的。所以造成Thread N阻塞。

第二个问题,为什么previousThread线程执行完毕就能够唤醒住线程呢?或者说是在什么时候唤醒的?

要了解这个问题,我们又得翻jdk的源码,但是如果大家对线程有一定的基本了解的话,通过wait方法阻塞的线程,需要通过notify或者notifyall来唤醒。所以在线程执行完毕以后会有一个唤醒的操作,只是我们不需要关心。
接下来在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 anywaythread->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()方法返回falsejava_lang_Thread::set_thread(threadObj(), NULL);lock.notify_all(thread);//注意这里// Ignore pending exception (ThreadDeath), since we are exiting anywaythread->clear_pending_exception();
}

ensure_join方法中,调用 lock.notify_all(thread); 唤醒所有等待thread锁的线程,意味着调用了join方法被阻塞的Thread N会被唤醒; 到目前为止,我们基本上对join的原理做了一个比较详细的分析

总结,Thread.join其实底层是通过wait/notifyall来实现线程的通信达到线程阻塞的目的;当线程执行结束以后,会触发两个事情,第一个是设置native线程对象为null、第二个是通过notifyall方法,让等待在previousThread对象锁上的wait方法被唤醒。

什么时候会使用Thread.join

在实际应用开发中,我们很少会使用thread.join。在实际使用过程中,我们可以通过join方法来等待线程执行的结果,其实有点类似future/callable的功能。
我们通过以下伪代码来说明join的使用场景

public void joinDemo(){//....Thread t=new Thread(payService);t.start();//.... //其他业务逻辑处理,不需要确定t线程是否执行完insertData();//后续的处理,需要依赖t线程的执行结果,可以在这里调用join方法等待t线程执行结束t.join();
}

Thread.join的作用和原理相关推荐

  1. Java thread.join

    thread.join()的作用是让当前线程等待线程thread终止.当前线程在join的过程中,如果被interrupt了,则join()方法抛出InterruptedException. 当调用j ...

  2. java并发编程之thread.join()方法详解

    thread.join()方法的作用:保证线程的执行结果的可见性.原理是通过阻塞主线程实现的. 代码Demo如下: public class ThreadJoinDemo {public static ...

  3. Java Thread.join()详解

     一.使用方式. join是Thread类的一个方法,启动线程后直接调用,例如: Thread t = new AThread(); t.start(); t.join(); 二.为什么要用joi ...

  4. Thread系列之Thread.Join()

    Thread.Join()方法,顾名思义,在一个线程中加入一些东西. MSDN上解释其作用为:阻塞 "调用线程" 直到某个线程结束. 这个翻译过来的解释有点晦涩.举个例子如下: s ...

  5. java join 异常_Java:守护进程:thread.join()没有完成,当在一个线程中抛出异常时...

    我写了一个Java守护进程(一个实现守护进程和Runnable的类),现在我遇到了以下问题: 在init()中,我创建了一个新线程 . Thread thread = new Thread(this) ...

  6. Thread.join(), CountDownLatch、CyclicBarrier和 Semaphore区别,联系及应用

    在java 1.5中,提供了一些非常有用的辅助类来帮助我们进行并发编程,比如CountDownLatch,CyclicBarrier和Semaphore,今天我们就来学习一下这三个辅助类的用法, 由于 ...

  7. java thread join()_Java中Thread.join()的使用方法

    概要 本文分三个部分对thread.join()进行分析: 1. join() 的示例和作用 2. join() 源码分析 3. 对网上其他分析 join() 的文章提出疑问 1. join() 的示 ...

  8. 主从复制面试之作用和原理

    主从复制面试之作用和原理 有些同学连集群和主从都分不清楚的,这里我说一下他们最本质的区别,其实也就是data-sharing和nothing-sharing的区别.集群是共享存储的.主从复制中没有任何 ...

  9. join()方法作用

    join作用是让其他线程变为等待,    t1.join();// 让其他线程变为等待,直到当前t1线程执行完毕,才释放. thread.Join把指定的线程加入到当前线程,可以将两个交替执行的线程合 ...

最新文章

  1. oracle 优化之批量处理bulk correct 和 forall
  2. MySQL主从复制-双主结构
  3. Python文件的多种读写方式及游标
  4. 励遍图块中的实体_如何删除CAD中的顽固图层?【AutoCAD教程】
  5. 做网页前端遇到的一些问题
  6. 次世代的会话管理项目 Spring Session
  7. Ubuntu 安装arm-linux-gcc交叉编译工具
  8. python3 读写json文件,python3没有读取JSON文件righ
  9. java栈属于哪个代,Java 代码执行原理
  10. Rust: trim(),trim_matches()等江南六怪......
  11. 快速使用ros小乌龟教程——ROS初体验
  12. Markdown部分语法使用
  13. PM 时间管理技能—麦肯锡30秒电梯理论
  14. 华为 eNSP 模拟器配置动态路由(ospf)
  15. P3406 海底高铁(前缀和+差分+坑点)
  16. 关于电阻的介绍,以及两线法和四线法测电阻的原理
  17. SC92F8003读24C64程序
  18. Spring Cloud 项目实战 Jenkins 实现 CI/CD 你需要的这里都有
  19. arcgis不闭合线转面_ArcGIS不闭合线转面
  20. PyTorch-Kaldi 深度学习语音识别开源软件

热门文章

  1. Android之使用AlertDialog类和AlertDialog.Builder类创建带取消,确定,中立的对话框
  2. java把date改成时间戳_spring处理数据库中Date类型字段转换成时间戳问题
  3. css3新增特性集合贴
  4. 如何借助配置中心ACM加速企业IT服务快速迭代
  5. ARM linux的启动部分源代码简略分析【转】
  6. Silverlight4中右键菜单实现-附源码下载
  7. mysqlsla安装与慢查询分析
  8. iText操作Word工具类
  9. SQL Servr 2008空间数据应用系列一:空间信息基础
  10. (转载)查看Oracle字符集及怎样修改字符集