join()——化多线程并发执行为顺序执行

多线程并发比较常见,但很多时候一个线程的输入可能非常依赖于另外一个线程的输出,此时这个线程就需要等待依赖线程执行完毕再执行。通俗的说,就是合并多个路径为单一路径,变多线程并发为单线程顺序执行。JDK提供了Join()这个Thread类的实例方法来达到这个要求。

join——英文含义为加入,通俗的说就是将一个线程合并到另一个线程。但是还有以下几个问题:

哪个线程加入?

加入到哪个线程?

加入后的效果是啥?

通过下面一段代码可以进行验证:

public class JoinTest extends Thread {
    private static int i = 0;

    @Override
    public void run() {
        while (i < 100000) {
           i++;
        }
    }

    public static void main(String[] args) throws Exception {
        Thread th = new JoinTest();
        //注意:必须先start,再join
        th.start();
        th.join();
        System.out.println(i);
    }
}

       执行结果为:100000       

 验证后可以回答上面三个问题了。

       哪个线程加入?——>调用Join()函数的那个对象开启的线程,注意要先开启线程,才能加入start,也就是start()方法要在join()方法前面。本例中是

JoinTest

     加入到哪个线程?——>加入到当前线程,通俗的说在哪个线程中执行该函数的,就加入哪个线程。

本例中是

main

    加入后的效果?——>阻塞当前线程,直到加入的线程执行完,当前线程继续执行。

    join的本质是让当前线程wait()在调用join方法的那个线程实例的锁上,等到那个线程实例开启的线程跑完,再notify()当前线程继续执行。下面是JDK中join()实现的核心代码片段:

public final synchronized void join(long paramLong)
			throws InterruptedException {
		long l1 = System.currentTimeMillis();
		long l2 = 0L;
		if (paramLong < 0L)
			throw new IllegalArgumentException("timeout value is negative");
	        //过期时间为0
		if (paramLong == 0L)
			while (true) {
                                //isAlive()是一个实例方法,用来判断线程(不是当前main线程,
                                //而是含有该方法的线程实例JoinTest启动的那个线程)是否存活,通俗的说就是run方法有没有跑完
				if (!(isAlive()))
					return;
				//这里是关键,同步方法内调用锁对象的wait()方法阻塞当前线程
				//可能这里会有疑问,阻塞的难道不是JoinTest线程吗?你看,是JoinTest对象里的方法
				//是不是JoinTest线程要看它在不在run方法里面,这个函数在run方法里面吗?显然不是
				super.wait(0L);//被合并线程未结束,当前线程继续等待
			}
		while (isAlive()) {
			long l3 = paramLong - l2;
			if (l3 <= 0L)
				return;
			super.wait(l3);
			l2 = System.currentTimeMillis() - l1;
		}
	}

    之后的流程我们也能猜到,被合并的线程执行完后,会notify()【其实这里是notifyall()】等待线程继续执行。



转载于:https://my.oschina.net/u/2529036/blog/618014

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值