java join

join:主线程等待子线程执行完,再继续执行主线程。

例子如下:

    public class JoinTest implements Runnable{  
          
        public static int a = 0;  
      
        public void run() {  
            for (int k = 0; k < 5; k++) {  
                a = a + 1;  
            }  
        }  
      
        public static void main(String[] args) throws Exception {  
            Runnable r = new JoinTest();  
            Thread t = new Thread(r);  
            t.start();        
            System.out.println(a);  
        }         
    }  

请问程序的输出结果是5吗?答案是:有可能。其实你很难遇到输出5的时候,通常情况下都不是5。当然这也和机器有严重的关系。为什么呢?我的解释是当主线程main方法执行System.out.println(a);这条语句时,线程还没有真正开始运行,或许正在为它分配资源准备运行。因为为线程分配资源需要时间,而main方法执行完t.start()方法后继续往下执行System.out.println(a);,这个时候得到的结果是a还没有被改变的值0。怎样才能让输出结果为5!其实很简单,join() 方法提供了这种功能。join() 方法,它能够使调用该方法的线程在此之前执行完毕。

public static void main(String[] args) throws Exception {  
        Runnable r = new JoinTest();  
        Thread t = new Thread(r);  
        t.start();        
        t.join(); //加入join()  
        System.out.println(a);  
    }  

这个时候,输出为5。



join的专业解释为:

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

主线程生成并起动了子线程,而子线程里要进行大量的耗时的运算(这里可以借鉴下线程的作用),当主线程处理完其他的事务后,需要用到子线程的处理结果,这个时候就要用到join();方法了。



需要注意的有:

1:如果线程被生成了,但还未被起动,调用它的join()方法是没有作用的,将直接继续向下执行

2:当main线程调用t.join时候,main线程会获得线程对象t的锁(wait 意味着拿到该对象的锁),调用该对象的wait(等待时间),直到该对象唤醒main线程

3:如果使用t.join(1000);那么主线程只等待1000ms,不管子线程什么时候结束。



main 线程调用t.join时,必须能够拿到线程t对象的锁,如果拿不到它是无法wait的,刚开的例子t.join(1000)不是说明了main线程等待1秒,如果在它等待之前,其他线程获取了t对象的锁,它等待时间可不就是1毫秒了

例子:

    class RunnableImpl implements Runnable {  
      
        public void run() {  
            try {  
                System.out.println("Begin sleep");  
                Thread.sleep(2000);  
                System.out.println("End sleep");  
            } catch (InterruptedException e) {  
                e.printStackTrace();  
            }  
        }  
    }  

    class ThreadTest extends Thread {  
      
        Thread thread;  
      
        public ThreadTest(Thread thread) {  
            this.thread = thread;  
        }  
      
        @Override  
        public void run() {  
            synchronized (thread) {  
                System.out.println("getObjectLock");  
                try {  
                    Thread.sleep(9000);  
                } catch (InterruptedException ex) {  
                 ex.printStackTrace();  
                }  
                System.out.println("ReleaseObjectLock");  
            }  
        }  
    }  

    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) {  
                    e.printStackTrace();           
                }  
            }  
    }  
输出:

getObjectLock
Begin sleep
End sleep
ReleaseObjectLock
joinFinish

 

在main方法中 通过new ThreadTest(t).start()实例化ThreadTest 线程对象, 它通过 synchronized (thread),获取线程对象t的锁,并Sleep(9000)后释放,这就意味着,即使main方法t.join(1000)等待一秒钟,它必须等待ThreadTest 线程释放t锁后才能进入wait方法中,它实际等待时间是9000+1000ms。


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值