join()是Thread类的一个方法。根据jdk文档的定义:
public final void join()throws InterruptedException: Waits for this thread to die. join()方法的作用,是等待这个线程结束
但显然,这样的定义并不清晰。个人认为"Java 7 Concurrency Cookbook"的定义较为清晰:
join() method suspends the execution of the calling thread until the object called finishes its execution.
也就是说,t.join()方法阻塞调用此方法的线程(calling thread),直到线程t完成,此线程再继续;通常用于在main()主线程内,等待其它线程完成再结束main()主线程。例如:
package com.bijian.test; import java.util.Date; import java.util.concurrent.TimeUnit; public class JoinTester01 implements Runnable { private String name; public JoinTester01(String name) { this.name = name; } public void run() { System.out.printf("%s begins: %s\n", name, new Date()); try { TimeUnit.SECONDS.sleep(4); } catch (InterruptedException e) { e.printStackTrace(); } System.out.printf("%s has finished: %s\n", name, new Date()); } public static void main(String[] args) { Thread thread1 = new Thread(new JoinTester01("One")); Thread thread2 = new Thread(new JoinTester01("Two")); thread1.start(); thread2.start(); try { thread1.join(); thread2.join(); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println("Main thread is finished"); } }
上述代码如果没有join()方法,输出如下:
Main thread is finished One begins: Wed Aug 28 10:21:36 CST 2013 Two begins: Wed Aug 28 10:21:36 CST 2013 Two has finished: Wed Aug 28 10:21:40 CST 2013 One has finished: Wed Aug 28 10:21:40 CST 2013
但有了join()方法,输出如下:
One begins: Sun Jan 20 17:53:38 CST 2019 Two begins: Sun Jan 20 17:53:38 CST 2019 Two has finished: Sun Jan 20 17:53:42 CST 2019 One has finished: Sun Jan 20 17:53:42 CST 2019 Main thread is finished
可以看出主线程main比其它两个线程先结束。
最后来深入了解一下join(),请看其源码:
/** * Waits at most {@code millis} milliseconds for this thread to * die. A timeout of {@code 0} means to wait forever. * * <p> This implementation uses a loop of {@code this.wait} calls * conditioned on {@code this.isAlive}. As a thread terminates the * {@code this.notifyAll} method is invoked. It is recommended that * applications not use {@code wait}, {@code notify}, or * {@code notifyAll} on {@code Thread} instances. * * @param millis * the time to wait in milliseconds * * @throws IllegalArgumentException * if the value of {@code millis} is negative * * @throws InterruptedException * if any thread has interrupted the current thread. The * <i>interrupted status</i> of the current thread is * cleared when this exception is thrown. * 此处A timeout of 0 means to wait forever 字面意思是永远等待,其实是等到t结束后 */ 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); } } else { while (isAlive()) { long delay = millis - now; if (delay <= 0) { break; } wait(delay); now = System.currentTimeMillis() - base; } } }
可以看出,Join方法实现是通过wait(小提示:Object 提供的方法)。 当main线程调用t.join时候,main线程会获得线程对象t的锁(wait 意味着拿到该对象的锁),调用该对象的wait(等待时间),直到该对象唤醒main线程 ,比如退出后。这就意味着main 线程调用t.join时,必须能够拿到线程t对象的锁。
package com.bijian.test; public class JoinTester02 implements Runnable { Thread thread; public JoinTester02(Thread thread) { this.thread = thread; } public void run() { synchronized (thread) { System.out.println("getObjectLock"); try { Thread.sleep(9000); } catch (InterruptedException ex) { ex.printStackTrace(); } System.out.println("ReleaseObjectLock"); } } public static void main(String[] args) { Thread thread = new Thread(new JoinTester01("Three")); Thread getLockThread = new Thread(new JoinTester02(thread)); getLockThread.start(); thread.start(); try { thread.join(); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println("Main finished!"); } }
输出如下:
getObjectLock Three begins: Sun Jan 20 18:05:31 CST 2019 Three has finished: Sun Jan 20 18:05:35 CST 2019 ReleaseObjectLock Main finished!
getLockThread通过 synchronized(thread),获取线程对象t的锁,并Sleep(9000)后释放,这就意味着,即使main方法t.join(1000)等待一秒钟,它必须等待ThreadTest 线程释放t锁后才能进入wait方法中。
相关推荐
在这个简谈中,我们将深入探讨`FutureTask`的实现原理和使用方式。 首先,`FutureTask`实现了`Runnable`接口,这意味着它可以直接被`Executor`服务执行。同时,它也实现了`Future`接口,提供了检查任务是否完成、...
Java中的`static`、`this`、`super`和`final`是四个非常重要的关键字,它们在编程中扮演着至关重要的角色。这篇文章将简要介绍这些关键字的基本用法及其应用场景。 1. **static** `static`关键字用于声明类级别的...
"班级管理方法简谈" 班级管理是学校教育中的一项重要工作,直接关系到学生的学习和成长。在《班级管理方法简谈》中,作者卢海战提出了五点班级管理方法,旨在提高班级管理的效率和质量。 首先,作者强调了加强学生...
JDK动态代理机制是Java反射机制的一个重要应用,它允许程序在运行时创建一个实现了特定接口的新类实例,并且能够控制这些新类实例的方法调用行为。这种机制不仅提高了代码的灵活性,而且为诸如AOP(面向切面编程)等...
简谈互联网时代高校辅导员的管理工作方法.pdf
JAVA面试题简谈你对synchronized关键字的理解 Synchronized关键字是Java语言中的一种同步机制,主要用于解决多线程之间的同步问题。下面我们将详细介绍synchronized关键字的特性、用法和注意事项。 Synchronized...
具体操作方法可以采用画布(Canvas)中的`drawImage`或`drawString`方法来调整图像位置,确保图像能够在较小的屏幕上正确显示。 2. **性能优化** J2ME设备通常具有较低的硬件配置,这对手游的性能提出了挑战。为了...
简谈Windows下的反调试技术 简谈Windows下的反调试技术 简谈Windows下的反调试技术 简谈Windows下的反调试技术 简谈Windows下的反调试技术 简谈Windows下的反调试技术
简谈计算机应用基础教学 简谈计算机应用基础教学 任务驱动教学法是一种建立在建构主义学习理论基础上的教学法,怎样分析计算 机应用基础教学? 一、引言 从事中职计算机教学多年来,发现了一个非常普遍的现象,即使...
简谈工程项目成本管理.doc
简谈公司员工绩效承诺.doc
房地产开发流程简谈.pptx
简谈英文自我介绍精选.doc
简谈ERP上机实验心得体会
计算机网络安全漏洞防范简谈.pdf
简谈校园网络安全方案的设计.pdf
欧柏泰克:.NET 简谈面向接口编程 面向接口编程是一种高抽象的开发模式,旨在将类与类之间的关系提升到一个更高的抽象层次。这种编程方式可以帮助开发人员更好地设计和实现软件系统,从而提高开发效率和质量。 在...
根据提供的文档信息,本文将详细解析注册表编程的基础知识及如何创建注册表补丁的方法。 ### 注册表编程概述 #### 注册表简介 注册表是Windows操作系统中用于存储系统设置、应用程序配置以及用户偏好设置的一个...
【手机成像技术简谈】 手机成像技术是现代生活中不可或缺的一部分,随着智能手机的发展,越来越多的人选择使用手机作为日常拍照的主要工具。手机成像技术的关键在于如何在各种环境条件下捕捉到理想亮度的照片,这...