join简介
join()方法是Thread类中的一个方法,该方法的定义是等待该线程终止。其实就是join()方法将挂起调用线程的执行,直到被调用的对象完成它的执行。
join实例
现在有T1、T2、T3
三个线程,你怎样保证T2
在T1
执行完后执行,T3
在T2
执行完后执行?这个问题是网上很热门的面试题目(这里除了用join
之外还有很多其他方法能够实现,只是使用join是最简单的方案),下面是实现的代码:
/**
* 现在有T1、T2、T3三个线程,你怎样保证T2在T1执行完后执行,T3在T2执行完后执行?
* @author RJH
* 2017年11月24日
*/
public class JoinDemo {
public static void main(String[] args) {
//初始化线程t1,由于后续有匿名内部类调用这个对象,需要用final修饰
final Thread t1 = new Thread(new Runnable() {
@Override
public void run() {
System.out.println("t1 is running");
}
});
//初始化线程t2,由于后续有匿名内部类调用这个对象,需要用final修饰
final Thread t2 = new Thread(new Runnable() {
@Override
public void run() {
try {
//t1调用join方法,t2会等待t1运行完之后才会开始执行后续代码
t1.join();
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
System.out.println("t2 is running");
}
}
});
//初始化线程t3
Thread t3 = new Thread(new Runnable() {
@Override
public void run() {
try {
//t2调用join方法,t3会等待t2运行完之后才会开始执行后续代码
t2.join();
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
System.out.println("t3 is running");
}
}
});
//依次启动3个线程
t1.start();
t2.start();
t3.start();
}
}
输出结果
t1 is running
t2 is running
t3 is running
结果分析
在t2线程中t2本身就是调用线程,所谓的调用线程是指调用了t.join()方法的线程,而被调用的对象指的是调用join方法的线程对象,即t1。所以这3个线程按照t1->t2->t3的顺序执行了。
CountDownnLatch
有两个线程,第一个线程需要等待第二个线程执行完所有的任务之后,第一个线程才能继续执行,那咱们可以使用线程中的哪个机制能实现呢?很显然利用Thread.join()方法,这是人人皆知的,但是还有一种场景是用Thread.join()解决不了的,具体啥场景,下面开始瞅一下,肯定是需要用到CoutDownLatch这个类来解决了。
先来看一个场景:
启动了一个主服务,然后它下面会启动五个子服务,而当五个子服务都执行到一半的逻辑时【或者是特定条件成立时】则通知主服务可以继续往下执行了,而五个子服务没有执行到一半完成之前主服务只能等待。
很明显这种业务场景用Thread.join()是没法实现的,因为join()是一定要等待指定线程都执行消亡了调用方才能继续往下执行
1 概念
CountDownLatch
允许一个或者多个线程去等待其他线程完成操作。
CountDownLatch
接收一个int
型参数,表示要等待的工作线程的个数。
当然也不一定是多线程,在单线程中可以用这个int
型参数表示多个操作步骤。
2 方法
CountDownLatch 提供了一些方法:
方法 | 说明 |
await() | 使当前线程进入同步队列进行等待,直到latch 的值被减到0 或者当前线程被中断,当前线程就会被唤醒。 |
await(long timeout, TimeUnit unit) | 带超时时间的await() |
countDown() | 使 上的线程 |
getCount() | 获得latch 的数值 |