等待线程结束
- 准备工作
- 实现过程
- 工作原理
- 扩展学习
可以用线程来初始化任务,直到初始化结束后,再执行程序的后续任务。
通过使用Thread类中的join()方法,可以达到此目的。当使用线程对象调用join()方法时,它延缓运行此线程,直到对象执行结束。
在本节中,通过一个初始化范例学习使用这个方法。
准备工作
本范例通过Eclipse开发工具实现。如果使用诸如NetBeans的开发工具,打开并创建一个新的Java项目。
实现过程
通过如下步骤完成范例:
- 创建DataSourcesLoader类,指定其实现Runnable接口:
public class DataSourcesLoader implements Runnable {
- 实现run()方法。通过输出一条表明其开始执行的信息,休眠4秒钟后,再输出一条表明结束执行的信息:
@Override
public void run() {
System.out.printf("Beginning data sources loading: %s\n", new Date());
try {
TimeUnit.SECONDS.sleep(4);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.printf("Data sources loading has finished: %s\n", new Date());
}
- 创建NetworkConnectionsLoader类,指定其实现Runnable接口。实现run()方法,内容与DataSourcesLoader类相同,但是休眠时间为6秒钟 。
- 现在实现主类,包含main()方法的Main类:
public class Main {
public static void main(String[] args) {
- 创建DataSourcesLoader类的对象以及运行此对象的线程:
DataSourcesLoader dsLoader = new DataSourcesLoader();
Thread thread1 = new Thread(dsLoader, "DataSourcesLoader");
- 创建NetworkConnectionsLoader类的对象以及运行此对象的线程:
NetworkConnectionsLoader ncLoader = new NetworkConnectionsLoader();
Thread thread2 = new Thread(ncLoader, "NetworkConnectionsLoader");
- 调用start()方法执行两个线程:
thread1.start();
thread2.start();
- 使用join()方法等待两个线程运行结束。这个方法能够抛出InterruptedException异常,所以我们使用代码捕获它:
try {
thread1.join();
thread2.join();
} catch (InterruptedException e) {
e.printStackTrace();
}
- 在控制台中输出表明程序结束的信息:
System.out.printf("Main: Configuration has been loaded: %s\n", new Date());
- 运行程序,查看结果。
工作原理
当运行范例时,需要理解两个线程对象如何开始执行的。首先DataSourcesLoader线程运行结束,然后NetworkConnectionsLoader类运行完成。同时,main线程对象才继续执行,输出最后的信息。
扩展学习
Java提供join()方法的另外两种方式:
- join(long milliseconds)
- join(long milliseconds, long nanos)
第一种join()方法代替不确定的等待调用线程结束时间,这个方法的参数将作为调用线程等待的毫秒数。例如,thread1对象中包含thread2.join(1000)指令,thread1将延缓执行,直到以下情形中的任何一种情况发生:
- thread2已经执行结束
- 经过了1000毫秒
当其中一种情况是true,join()方法返回。通过检测线程的状态获知是否返回join()方法,因为线程已经执行结束或者通过规定的时间。
第二种join()方法与第一种相似,不过它接收参数为毫秒和纳秒两个数值。