`
kankan1218
  • 浏览: 271863 次
  • 性别: Icon_minigender_1
  • 来自: 大连
社区版块
存档分类
最新评论

java join方法详解

阅读更多

 

    方法Join 是干啥用的? 简单回答,同步,如何同步? 怎么实现的? 下面将逐个回答。

   自从接触Java 多线程,一直对Join 理解不了。JDK 是这样说的:join public final void joinlong millisthrows InterruptedException Waits at most millis milliseconds for this thread to die. A timeout of 0 means to wait forever. 大家能理解吗? 字面意思是等待一段时间直到这个线程死亡,我的疑问是那个线程,是它本身的线程还是调用它的线程的,上代码: 

package  concurrentstudy;
/**
 *
 * 
@author  vma
 */

public   class  JoinTest {
    
public   static   void  main(String[] args) {
        Thread t = 
new  Thread( new  RunnableImpl());
        t.start();
        
try  {
          
  t.join(1000) ; // 主线程只等1 秒,不管子线程什么时候结束
            System.out.println("joinFinish");
        } 
catch  (InterruptedException e) {
            
// TODO Auto-generated catch block
            e.printStackTrace();
     
        }
    }
}
class  RunnableImpl  implements  Runnable {

    @Override
    
public   void  run() {
        
try  {
            System.out.println("Begin sleep");
            Thread.sleep(1000);
           System.out.println("End sleep");
        } 
catch  (InterruptedException e) {
            e.printStackTrace();
        }

    }
}

结果是:
Begin sleep
End sleep
joinFinish
明白了吧, main 线程调用t.join 时,main 线程等待t 线程 ,等待时间是1000 ,如果t 线程Sleep 2000 
  public   void  run() {
         try  {
            System.out.println("Begin sleep");
            // Thread.sleep(1000);
            Thread.sleep(2000) ;
           System.out.println("End sleep");
        }  catch  (InterruptedException e) {
            e.printStackTrace();
        }

    }
结果是:
Begin sleep
joinFinish
End sleep
也就是说 main 线程只等1000 毫秒,不管T 什么时候结束 ,如果是t.join() 呢, 看代码:   
 public final void join() throws InterruptedException {
    join(0);
    }
    就是说如果是t.join() = t.join(0)  JDK 这样说的 A timeout of  0  means to wait forever 字面意思是永远等待,是这样吗?
    其实是等到t 结束后。
    这个是怎么实现的吗? 看JDK 代码:

     /**
     * Waits at most <code>millis</code> milliseconds for this thread to 
     * die. A timeout of <code>0</code> means to wait forever. 
     *
     * 
@param       millis   the time to wait in milliseconds.
     * 
@exception   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.
     */

    
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 对象的锁 ,如果拿不到它是无法wait 的,刚开的例子t.join(1000) 不是说明了main 线程等待1 秒,如果在它等待之前,其他线程获取了t 对象的锁,它等待时间可不就是1 毫秒了。上代码介绍:

/*
 * To change this template, choose Tools | Templates
 * and open the template in the editor.
 */

package  concurrentstudy;
/**
 *
 * 
@author  vma
 */

public   class  JoinTest {
    
public   static   void  main(String[] args) {
        Thread t = 
new  Thread( new  RunnableImpl());
       
new  ThreadTest(t).start();// 这个线程会持有锁
        t.start();
        
try  {
            t.join();
            System.out.println("joinFinish");
        } 
catch  (InterruptedException e) {
            
// TODO Auto-generated catch block
            e.printStackTrace();
     
        }
    }
}
class  ThreadTest  extends  Thread {

    Thread thread;

    
public  ThreadTest(Thread thread) {
        
this .thread = thread;
    }

    @Override
    
public   void  run() {
        holdThreadLock();
    }

    
public   void  holdThreadLock() {
        
synchronized  (thread) {
            System.out.println("getObjectLock");
            
try  {
                Thread.sleep(9000);

            } 
catch  (InterruptedException ex) {
             ex.printStackTrace();
            }
            System.out.println("ReleaseObjectLock");
        }

    }
}

class  RunnableImpl  implements  Runnable {

    @Override
    
public   void  run() {
        
try  {
            System.out.println("Begin sleep");
            Thread.sleep(2000);
           System.out.println("End sleep");
        } 
catch  (InterruptedException e) {
            e.printStackTrace();
        }


    }
}

      main 方法中   通过new  ThreadTest(t).start(); 实例化ThreadTest   线程对象, 它在 holdThreadLock() 方法中,通过   synchronized  (thread) ,获取线程对象t 的锁,并Sleep9000 )后释放,这就意味着,即使
main
方法t.join(1000), 等待一秒钟,它必须等待
ThreadTest   线程释放t 锁后才能进入wait 方法中,它实际等待时间是9000+1000 MS
运行结果是:
getObjectLock
Begin sleep
End sleep
ReleaseObjectLock
joinFinish

 

分享到:
评论

相关推荐

    JAVA多线程之方法 JOIN详解及实例代码

    主要介绍了JAVA多线程之方法 JOIN详解及实例代码的相关资料,需要的朋友可以参考下

    java基本教程之join方法详解 java多线程教程

    本文对java Thread中join()方法进行介绍,join()的作用是让“主线程”等待“子线程”结束之后才能继续运行,大家参考使用吧

    Mysql之innerjoin,leftjoin,rightjoin详解.pdf

    Mysql之innerjoin,leftjoin,rightjoin详解.pdf

    JAVA WEB 开发详解:XML+XSLT+SERVLET+JSP 深入剖析与实例应用.part2

    6.6.4 sendredirect()和forward()方法的区别 238 6.7 小结 239 第7章 web应用程序的部署 240 7.1 配置任意目录下的web应用程序 240 7.2 war文件 242 7.3 tomcat中servlet的另一种运行方式 244 7.4 与servlet...

    JAVA WEB 开发详解:XML+XSLT+SERVLET+JSP 深入剖析与实例应用.part3

    6.6.4 sendredirect()和forward()方法的区别 238 6.7 小结 239 第7章 web应用程序的部署 240 7.1 配置任意目录下的web应用程序 240 7.2 war文件 242 7.3 tomcat中servlet的另一种运行方式 244 7.4 与servlet...

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

    主要介绍了java 中Thread.join()的使用方法的相关资料,需要的朋友可以参考下

    JAVA WEB 开发详解:XML+XSLT+SERVLET+JSP 深入剖析与实例应用.part4

    6.6.4 sendredirect()和forward()方法的区别 238 6.7 小结 239 第7章 web应用程序的部署 240 7.1 配置任意目录下的web应用程序 240 7.2 war文件 242 7.3 tomcat中servlet的另一种运行方式 244 7.4 与servlet...

    JAVA多线程编程详解-详细操作例子

    本压缩包,总共包含两个文档,JAVA多线程编程详解-详细操作例子和 Java多线 程编程总结 例如,runnable、thread、stop()、 suspend、yield、setPriority()、getPriority()、synchronized、wait()、join、线程池同步...

    java8 lambda详解

    上图中java定义个一个回调接口ActionListener,通过该接口来指定监听之后按钮的行为并添加相应操作。先看图一咯,一般我们监听一个按钮是用匿名内部类的形式,而不会专门去创建一个implement ActionListener的类,...

    JAVA WEB 开发详解:XML+XSLT+SERVLET+JSP 深入剖析与实例应用.part5

    6.6.4 sendredirect()和forward()方法的区别 238 6.7 小结 239 第7章 web应用程序的部署 240 7.1 配置任意目录下的web应用程序 240 7.2 war文件 242 7.3 tomcat中servlet的另一种运行方式 244 7.4 与servlet...

    MySQL中的JOIN详解及sql实战

    用途:帮助读者了解MySQL中的JOIN操作,掌握LEFT JOIN、ON条件过滤和笛卡尔积的使用方法,从而在实际开发中灵活运用JOIN操作实现表关联查询。本文还提供了扩展内容,如排名等应用场景,帮助读者进一步理解JOIN操作的...

    Java线程编程中isAlive()和join()的使用详解

    主要介绍了Java线程编程中isAlive()和join()的使用详解,是Java入门学习中的基础知识,需要的朋友可以参考下

    精通 Hibernate:Java 对象持久化技术详解(第2版).part2

    第1章 Java应用分层架构及软件模型  1.1 应用程序的分层体系结构  1.1.1 区分物理层和逻辑层  1.1.2 软件层的特征  1.1.3 软件分层的优点  1.1.4 软件分层的缺点  1.1.5 Java应用的持久化层  1.2 软件的模型 ...

    java中List集合及其实现类的方法详解

    本篇文章给大家带来的内容是关于java中List集合及其实现类的方法介绍(附代码),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。下面我们就来学习一下吧

    JAVA高质量并发详解,多线程并发深入讲解

    接着,深入讲解了Java并发编程的核心API,如synchronized关键字、Lock接口、Condition接口、Semaphore等,帮助读者掌握Java并发编程的基本工具和方法。 除了基础知识和API的讲解,本书还重点介绍了Java并发编程的...

    Fork Join框架机制详解.docx

    Java提供Fork/Join框架用于并行执行任务,核心的思想就是将一个大任务切分成多个小任务,然后汇总每个小任务的执行结果得到这个大任务的最终结果。 这种机制策略在分布式数据库中非常常见,数据分布在不同的数据库的...

    疯狂JAVA讲义

    5.2 方法详解 116 5.2.1 方法的所属性 116 5.2.2 方法的参数传递机制 116 5.2.3 形参长度可变的方法 120 5.2.4 递归方法 121 5.2.5 方法重载 123 学生提问:为什么方法的返回值类型不能用于区分重载的方法? ...

    Java多线程详解

    文章目录1、进程与线程2、创建多线程2.1、继承Thread类2.2、实现Runnable接口2.3、使用匿名内部类实现2.4、实现Runnable接口的好处2.5、使用Callable和Future创建线程3、线程的生命周期4、几种特殊线程4.1、join线程...

Global site tag (gtag.js) - Google Analytics