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()主线程。例如:

  1. public class JoinTester01 implements Runnable {
  2.  
  3. private String name;
  4.  
  5. public JoinTester01(String name) {
  6. this.name = name;
  7. }
  8.  
  9. public void run() {
  10. System.out.printf("%s begins: %s\n", name, new Date());
  11. try {
  12. TimeUnit.SECONDS.sleep(4);
  13. } catch (InterruptedException e) {
  14. e.printStackTrace();
  15. }
  16. System.out.printf("%s has finished: %s\n", name, new Date());
  17. }
  18.  
  19. public static void main(String[] args) {
  20. Thread thread1 = new Thread(new JoinTester01("One"));
  21. Thread thread2 = new Thread(new JoinTester01("Two"));
  22. thread1.start();
  23. thread2.start();
  24.  
  25. try {
  26. thread1.join();
  27. thread2.join();
  28. } catch (InterruptedException e) {
  29. // TODO Auto-generated catch block
  30. e.printStackTrace();
  31. }
  32.  
  33. System.out.println("Main thread is finished");
  34. }
  35.  
  36. }

上述代码如果没有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

可以看出主线程main比其它两个线程先结束。

最后来深入了解一下join(),请看其源码:

  1. /**
  2. * Waits at most <code>millis</code> milliseconds for this thread to
  3. * die. A timeout of <code>0</code> means to wait forever.
  4. */
  5. //此处A timeout of 0 means to wait forever 字面意思是永远等待,其实是等到t结束后。
  6. public final synchronized void join(long millis) throws InterruptedException {
  7. long base = System.currentTimeMillis();
  8. long now = 0;
  9.  
  10. if (millis < 0) {
  11. throw new IllegalArgumentException("timeout value is negative");
  12. }
  13.  
  14. if (millis == 0) {
  15. while (isAlive()) {
  16. wait(0);
  17. }
  18. } else {
  19. while (isAlive()) {
  20. long delay = millis - now;
  21. if (delay <= 0) {
  22. break;
  23. }
  24. wait(delay);
  25. now = System.currentTimeMillis() - base;
  26. }
  27. }
  28. }

可以看出,Join方法实现是通过wait(小提示:Object 提供的方法)。 当main线程调用t.join时候,main线程会获得线程对象t的锁(wait 意味着拿到该对象的锁),调用该对象的wait(等待时间),直到该对象唤醒main线程 ,比如退出后。这就意味着main 线程调用t.join时,必须能够拿到线程t对象的锁。

  1. public class JoinTester02 implements Runnable {
  2.  
  3. Thread thread;
  4.  
  5. public JoinTester02(Thread thread) {
  6. this.thread = thread;
  7. }
  8.  
  9. public void run() {
  10. synchronized (thread) {
  11. System.out.println("getObjectLock");
  12. try {
  13. Thread.sleep(9000);
  14. } catch (InterruptedException ex) {
  15. ex.printStackTrace();
  16. }
  17. System.out.println("ReleaseObjectLock");
  18. }
  19. }
  20.  
  21. public static void main(String[] args) {
  22. Thread thread = new Thread(new JoinTester01("Three"));
  23. Thread getLockThread = new Thread(new JoinTester02(thread));
  24.  
  25. getLockThread.start();
  26. thread.start();
  27.  
  28. try {
  29. thread.join();
  30. } catch (InterruptedException e) {
  31. // TODO Auto-generated catch block
  32. e.printStackTrace();
  33. }
  34. System.out.println("Main finished!");
  35. }
  36.  
  37. }public class JoinTester02 implements Runnable {
  38.  
  39. Thread thread;
  40.  
  41. public JoinTester02(Thread thread) {
  42. this.thread = thread;
  43. }
  44.  
  45. public void run() {
  46. synchronized (thread) {
  47. System.out.println("getObjectLock");
  48. try {
  49. Thread.sleep(9000);
  50. } catch (InterruptedException ex) {
  51. ex.printStackTrace();
  52. }
  53. System.out.println("ReleaseObjectLock");
  54. }
  55. }
  56.  
  57. public static void main(String[] args) {
  58. Thread thread = new Thread(new JoinTester01("Three"));
  59. Thread getLockThread = new Thread(new JoinTester02(thread));
  60.  
  61. getLockThread.start();
  62. thread.start();
  63.  
  64. try {
  65. thread.join();
  66. } catch (InterruptedException e) {
  67. // TODO Auto-generated catch block
  68. e.printStackTrace();
  69. }
  70. System.out.println("Main finished!");
  71. }
  72.  
  73. }

输出如下:

getObjectLock
Three begins: Wed Aug 28 10:42:00 CST 2013
Three has finished: Wed Aug 28 10:42:04 CST 2013
ReleaseObjectLock
Main finished!

getLockThread通过 synchronized  (thread) ,获取线程对象t的锁,并Sleep(9000)后释放,这就意味着,即使main方法t.join(1000)等待一秒钟,它必须等待ThreadTest 线程释放t锁后才能进入wait方法中。

本文完

参考:

http://uule.iteye.com/blog/1101994

Java 7 Concurency Cookbook

简谈Java的join()方法的更多相关文章

  1. 简谈Java的join()方法(转)

    join()是Thread类的一个方法.根据jdk文档的定义: public final void join()throws InterruptedException: Waits for this ...

  2. 简谈Java语言的继承

    Java语言的继承 这里简谈Java语言的三大特性之二——继承. Java语言的三大特性是循序渐进的.是有顺序性的,应该按照封装-->继承-->多态这样的顺序依次学习 继承的定义 百度百科 ...

  3. 简谈Java语言的封装

    简谈Java语言的封装 封装的定义 封装将复杂模块或系统的逻辑实现细节隐藏,让使用者只需要关心这个模块或系统怎么使用,而不用关心这个模块或系统是怎么实现的. 在面向对象的的编程中,我们一般通过接口来描 ...

  4. java多线程 join方法以及优先级方法

    /*join:当A线程执行到了B线程的.join()方法时,A就会等待.等B线程都执行完,A才会执行. join可以用来临时加入线程执行. 1.线程使用join方法,主线程就停下,等它执行完,那么如果 ...

  5. Java中join()方法的理解

    thread.Join把指定的线程加入到当前线程,可以将两个交替执行的线程合并为顺序执行的线程. 比如在线程B中调用了线程A的Join()方法,直到线程A执行完毕后,才会继续执行线程B. t.join ...

  6. Java通过join方法来暂停当前线程

    目标线程的join方法暂停当前线程,直到目前线程完成(从run()方法返回). Java代码: package Threads; import java.io.IOException; /** * C ...

  7. [java] java 线程join方法详解

    join方法的作用是使所属线程对象正常执行run方法,而对当前线程无限期阻塞,直到所属线程销毁后再执行当前线程的逻辑. 一.先看普通的无join方法NoJoin.java public class N ...

  8. Java多线程-join方法

    thread.Join把指定的线程加入到当前线程,可以将两个交替执行的线程合并为顺序执行的线程.比如在线程B中调用了线程A的Join()方法,直到线程A执行完毕后,才会继续执行线程B. 具体例子看链接 ...

  9. java线程join方法使用方法简介

    本博客简介介绍一下java线程的join方法,join方法是实现线程同步,可以将原本并行执行的多线程方法变成串行执行的 如图所示代码,是并行执行的 public class ThreadTest { ...

随机推荐

  1. 关于Assets.car素材问题

    最近在做自己的第一个App,由于全程都是自己一个人完成,所以原型设计.素材都得自己找,自己改.遇到了提取Assets.car中的素材的问题,通过网络找到了2中解决方法: themeEngine 使 用 ...

  2. [QualityCenter]设置工作流脚本-设置不同字段值关联不同列表

    需求:当选择A字段某个值时,设置B字段的列表值根据A字段的值来判断读取不同的列表值,如当运行省份的值已更改, 运行地区的选择列表将更改. 在脚本编辑器新建一个函数UserFuntion_Bug_Pro ...

  3. Effective Java 67 Avoid excessive synchronization

    Principle To avoid liveness and safety failures, never cede control to the client within a synchroni ...

  4. 最短路径之迪杰斯特拉(Dijkstra)算法

    迪杰斯特拉(Dijkstra)算法主要是针对没有负值的有向图,求解其中的单一起点到其他顶点的最短路径算法.本文主要总结迪杰斯特拉(Dijkstra)算法的原理和算法流程,最后通过程序实现在一个带权值的 ...

  5. grub条目示例

    https://wiki.archlinux.org/index.php/GRUB#Install_to_disk /boot/grub/menu.lst default=0 timeout=5 ti ...

  6. python正则表达式 小例几则

    会用到的语法 正则字符 释义 举例 + 前面元素至少出现一次 ab+:ab.abbbb 等 * 前面元素出现0次或多次 ab*:a.ab.abb 等 ? 匹配前面的一次或0次 Ab?: A.Ab 等 ...

  7. (三)openwrt主Makefile解析

    本周成胖子每周一博到了第四周^_^ 前言 主Makefile结构 顶层 第二层 尾记 前言 前一篇,我们大概描述了整个镜像文件的生成过程.本周我们来解析主Makefile,看看主要编译过程是怎么产生的 ...

  8. (二)我的Makefile学习冲动&&编译过程概述

    前言 一 年轻的冲动 二 学习曲线 1 Makefile基本语法 2 bash基础 3 world 三 编译过程概述 1 主机预装工具 2 编译host工具 3 编译交叉工具链 4 编译内核模块 5 ...

  9. EXCEL IF 函数 模糊查询

    A列都是产品名,比如衬衫,长袖衬衫,短袖衬衫,短裙,长裙 搜索A列的产品名,凡是含有“衬衫”的一律在B列对应行输出“衬衫”,凡是含有“裙”字的一律输出“裙子”在B列对应行,请教一下怎么写函数,本来用I ...

  10. django安装和卸载

    一.pip安装 安装: apt-get update apt-get install python-pip pip install django==1.4.20 卸载: pip uninstall d ...