Java Thread.join()详解(转)
(1)join方法是可以中断的
(2)在线程joiner在另一个线程t上调用t.join(),线程joiner将被挂起,直到线程t结束(即t.isAlive()返回为false)才恢复
package thread.join2;
class Sleeper extends Thread{
private int duration;
public Sleeper(String name,int sleepTime) {
super(name);
duration=sleepTime;
start();
}
@Override
public void run() {
try {
sleep(duration);
} catch (InterruptedException e) {
System.out.println(currentThread()+" was interrupted. isInterrupted():"+isInterrupted());
return;
}
System.out.println(currentThread()+" was awakened");
}
}
class Joiner extends Thread{
private Sleeper sleeper;
public Joiner(String name,Sleeper sleeper) {
super(name);
this.sleeper=sleeper;
start();
}
@Override
public void run() {
try {
sleeper.join();
} catch (InterruptedException e) {
System.out.println(currentThread()+" Interrupted.");
}
System.out.println(currentThread()+" join completed.");
}
}
public class JoinTest {
public static void main(String[] args) {
int sleepTime=1500;
Sleeper sleepy_sleeper=new Sleeper("sleepy_sleeper", sleepTime);
Sleeper grumpy_sleeper=new Sleeper("grumpy_sleeper", sleepTime);
Joiner dopey_joiner=new Joiner("dopey_joiner", sleepy_sleeper);
Joiner doc_joiner=new Joiner("doc_joiner", grumpy_sleeper);
grumpy_sleeper.interrupt();
}
}
Output:
Thread[grumpy_sleeper,5,main] was interrupted. isInterrupted():false
Thread[doc_joiner,5,main] join completed.
Thread[sleepy_sleeper,5,main] was awakened
Thread[dopey_joiner,5,main] join completed.
一、使用方式。
join是Thread类的一个方法,启动线程后直接调用,例如:
|
1
|
Thread t = new AThread(); t.start(); t.join(); |
二、为什么要用join()方法
在很多情况下,主线程生成并起动了子线程,如果子线程里要进行大量的耗时的运算,主线程往往将于子线程之前结束,但是如果主线程处理完其他的事务后,需要用到子线程的处理结果,也就是主线程需要等待子线程执行完成之后再结束,这个时候就要用到join()方法了。
三、join方法的作用
在JDk的API里对于join()方法是:
join
public final void join() throws InterruptedException Waits for this thread to die. Throws: InterruptedException - if any thread has interrupted the current thread. The interrupted status of the current thread is cleared when this exception is thrown.
即join()的作用是:“等待该线程终止”,这里需要理解的就是该线程是指的主线程等待子线程的终止。也就是在子线程调用了join()方法后面的代码,只有等到子线程结束了才能执行。
四、用实例来理解
写一个简单的例子来看一下join()的用法:
1.AThread 类
BThread类
TestDemo 类
12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152classBThreadextendsThread {publicBThread() {super("[BThread] Thread");};publicvoidrun() {String threadName = Thread.currentThread().getName();System.out.println(threadName +" start.");try{for(inti =0; i <5; i++) {System.out.println(threadName +" loop at "+ i);Thread.sleep(1000);}System.out.println(threadName +" end.");}catch(Exception e) {System.out.println("Exception from "+ threadName +".run");}}}classAThreadextendsThread {BThread bt;publicAThread(BThread bt) {super("[AThread] Thread");this.bt = bt;}publicvoidrun() {String threadName = Thread.currentThread().getName();System.out.println(threadName +" start.");try{bt.join();System.out.println(threadName +" end.");}catch(Exception e) {System.out.println("Exception from "+ threadName +".run");}}}publicclassTestDemo {publicstaticvoidmain(String[] args) {String threadName = Thread.currentThread().getName();System.out.println(threadName +" start.");BThread bt =newBThread();AThread at =newAThread(bt);try{bt.start();Thread.sleep(2000);at.start();at.join();}catch(Exception e) {System.out.println("Exception from main");}System.out.println(threadName +" end!");}}打印结果:
1234567891011main start.//主线程起动,因为调用了at.join(),要等到at结束了,此线程才能向下执行。[BThread] Thread start.[BThread] Thread loop at 0[BThread] Thread loop at 1[AThread] Thread start.//线程at启动,因为调用bt.join(),等到bt结束了才向下执行。[BThread] Thread loop at 2[BThread] Thread loop at 3[BThread] Thread loop at 4[BThread] Thread end.[AThread] Thread end.//线程AThread在bt.join();阻塞处起动,向下继续执行的结果main end!//线程AThread结束,此线程在at.join();阻塞处起动,向下继续执行的结果。修改一下代码:
1234567891011121314151617publicclassTestDemo {publicstaticvoidmain(String[] args) {String threadName = Thread.currentThread().getName();System.out.println(threadName +" start.");BThread bt =newBThread();AThread at =newAThread(bt);try{bt.start();Thread.sleep(2000);at.start();//at.join(); //在此处注释掉对join()的调用}catch(Exception e) {System.out.println("Exception from main");}System.out.println(threadName +" end!");}}打印结果:
123456789101112main start.//主线程起动,因为Thread.sleep(2000),主线程没有马上结束;[BThread] Thread start.//线程BThread起动[BThread] Thread loop at 0[BThread] Thread loop at 1main end!//在sleep两秒后主线程结束,AThread执行的bt.join();并不会影响到主线程。[AThread] Thread start.//线程at起动,因为调用了bt.join(),等到bt结束了,此线程才向下执行。[BThread] Thread loop at 2[BThread] Thread loop at 3[BThread] Thread loop at 4[BThread] Thread end.//线程BThread结束了[AThread] Thread end.//线程AThread在bt.join();阻塞处起动,向下继续执行的结果五、从源码看join()方法
在AThread的run方法里,执行了bt.join();,进入看一下它的JDK源码:
123publicfinalvoidjoin()throwsInterruptedException {join(0L);}然后进入join(0L)方法:
123456789101112131415161718192021publicfinalsynchronizedvoidjoin(longl)throwsInterruptedException{longl1 = System.currentTimeMillis();longl2 = 0L;if(l < 0L)thrownewIllegalArgumentException("timeout value is negative");if(l == 0L)for(; isAlive(); wait(0L));elsedo{if(!isAlive())break;longl3 = l - l2;if(l3 <= 0L)break;wait(l3);l2 = System.currentTimeMillis() - l1;}while(true);}单纯从代码上看: * 如果线程被生成了,但还未被起动,isAlive()将返回false,调用它的join()方法是没有作用的。将直接继续向下执行。 * 在AThread类中的run方法中,bt.join()是判断bt的active状态,如果bt的isActive()方法返回false,在bt.join(),这一点就不用阻塞了,可以继续向下进行了。从源码里看,wait方法中有参数,也就是不用唤醒谁,只是不再执行wait,向下继续执行而已。 * 在join()方法中,对于isAlive()和wait()方法的作用对象是个比较让人困惑的问题:
isAlive()方法的签名是:public final native boolean isAlive(),也就是说isAlive()是判断当前线程的状态,也就是bt的状态。
wait()方法在jdk文档中的解释如下:
Causes the current thread to wait until another thread invokes the notify() method or the notifyAll() method for this object. In other words, this method behaves exactly as if it simply performs the call wait(0).
The current thread must own this object's monitor. The thread releases ownership of this monitor and waits until another thread notifies threads waiting on this object's monitor to wake up either through a call to the notify method or the notifyAll method. The thread then waits until it can re-obtain ownership of the monitor and resumes execution.
在这里,当前线程指的是at。
http://www.open-open.com/lib/view/open1371741636171.html
Java Thread.join()详解(转)的更多相关文章
- 【转】Java Thread.join()详解
http://www.open-open.com/lib/view/open1371741636171.html 一.使用方式. join是Thread类的一个方法,启动线程后直接调用,例如: ? 1 ...
- Java Thread.join()详解--父线程等待子线程结束后再结束
目录(?)[+] 阅读目录 一.使用方式. 二.为什么要用join()方法 三.join方法的作用 join 四.用实例来理解 打印结果: 打印结果: 五.从源码看join()方法 join是Th ...
- Java Thread.join()详解
一.使用方式. 二.为什么要用join()方法 三.join方法的作用 join 四.用实例来理解 打印结果: 打印结果: 五.从源码看join()方法 一.使用方式. join是Thread类的 ...
- Java Thread.yield详解
这是Java中的一种线程让步方法,让Java中的线程从执行状态变成就绪状态,然后处理器再从就绪队列中挑选线程进行执行(优先级大的,被挑选的概率较大),这种转换也不确定,让或者不让都是取决与处理器,线程 ...
- Thread.join详解
/** * 如果某个线程在另一个线程t上调用t.join:那么此线程将被挂起,直到目标t线程的结束才恢复即t.isAlive返回为假 * * @date:2018年6月27日 * @author:zh ...
- [译]Java Thread join示例与详解
Java Thread join示例与详解 Java Thread join方法用来暂停当前线程直到join操作上的线程结束.java中有三个重载的join方法: public final void ...
- Java基础-进程与线程之Thread类详解
Java基础-进程与线程之Thread类详解 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.进程与线程的区别 简而言之:一个程序运行后至少有一个进程,一个进程中可以包含多个线程 ...
- java.lang.Thread类详解
java.lang.Thread类详解 一.前言 位于java.lang包下的Thread类是非常重要的线程类,它实现了Runnable接口,今天我们来学习一下Thread类,在学习Thread类之前 ...
- Java线程创建形式 Thread构造详解 多线程中篇(五)
Thread作为线程的抽象,Thread的实例用于描述线程,对线程的操纵,就是对Thread实例对象的管理与控制. 创建一个线程这个问题,也就转换为如何构造一个正确的Thread对象. 构造方法列表 ...
随机推荐
- codeforces 437C The Child and Toy
time limit per test 1 second memory limit per test 256 megabytes input standard input output standar ...
- PHP - 操作MySQL数据库
第16章 PHP操作MySQL 学习要点: 1.PHP连接到MySQL 2.增删改查 3.其他常用函数 如果你已经具有了使用PHP.SQL和MySQL的丰富经验,现在就可以把所有这些技术组合在一起.P ...
- 记一次在Tomcat部署项目后无法启动该项目的例子
在一次正常部署后,发现在Tomcat的管理中无法将该项目启动起来,并在Tomcat的日志文件localhost.log中发现以下错误信息: java.lang.NoSuchMethodError: o ...
- Flash, Flex, Air, Flashplayer之间的相互关系是什么?
著作权归作者所有.商业转载请联系作者获得授权,非商业转载请注明出处.作者:曾嵘链接:http://www.zhihu.com/question/20001256/answer/15565376来源:知 ...
- PS大头照的背景
给同事ps一个大头照,只是修改一下背景颜色,以前没有做过,这次算是小练习了一把,与大家分享一下.修改大头照背景重要的是如何选中背景的区域,如果用魔棒选择的话,头与背景的边处理的不好,说说我的处理方法吧 ...
- 学习算法-基数排序(radix sort)卡片分类(card sort) C++数组实现
基数排序称为卡片分类,这是一个比较早的时间越多,排名方法. 现代计算机出现之前,它已被用于排序老式打孔卡. 说下基数排序的思想.前面我有写一个桶式排序,基数排序的思想是桶式排序的推广. 桶式排序:ht ...
- 【linux】内核+文件系统下载到开发板
K开发 欢迎转载,转载时请保留作者信息,谢谢. 邮箱:tangzhongp@163.com 博客园地址:http://www.cnblogs.com/embedded-tzp Csdn博客地址:htt ...
- 【linux】开发环境说明
欢迎转载,转载时请保留作者信息,谢谢. 邮箱:tangzhongp@163.com 博客园地址:http://www.cnblogs.com/embedded-tzp Csdn博客地址:http:// ...
- Eclipse用法和技巧十七:覆盖父类方法
在学校里面学习java,遇到访问权限修饰符一直停留在public是公有的,外面可以访问:protected是对子类可见的,外部不可以访问:private仅在本类中可见.工作之后,接触到了java代码多 ...
- Not able to reset SmartRF04DD
今天在使用使用CC2540的时候,想下载个程序到CC2540底板上,结果出现Not able to reset SmartRF04DD的错误.如下图 经过一番摸索,最终是按下CCDEBUG上的rese ...