等待线程结束

  • 准备工作
  • 实现过程
  • 工作原理
  • 扩展学习

可以用线程来初始化任务,直到初始化结束后,再执行程序的后续任务。

通过使用Thread类中的join()方法,可以达到此目的。当使用线程对象调用join()方法时,它延缓运行此线程,直到对象执行结束。

在本节中,通过一个初始化范例学习使用这个方法。

准备工作

本范例通过Eclipse开发工具实现。如果使用诸如NetBeans的开发工具,打开并创建一个新的Java项目。

实现过程

通过如下步骤完成范例:

  1. 创建DataSourcesLoader类,指定其实现Runnable接口:
public class DataSourcesLoader implements Runnable {
  1. 实现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());
	}
  1. 创建NetworkConnectionsLoader类,指定其实现Runnable接口。实现run()方法,内容与DataSourcesLoader类相同,但是休眠时间为6秒钟 。
  2. 现在实现主类,包含main()方法的Main类:
public class Main {
	public static void main(String[] args) {
  1. 创建DataSourcesLoader类的对象以及运行此对象的线程:
DataSourcesLoader dsLoader = new DataSourcesLoader();
		Thread thread1 = new Thread(dsLoader, "DataSourcesLoader");
  1. 创建NetworkConnectionsLoader类的对象以及运行此对象的线程:
NetworkConnectionsLoader ncLoader = new NetworkConnectionsLoader();
		Thread thread2 = new Thread(ncLoader, "NetworkConnectionsLoader");
  1. 调用start()方法执行两个线程:
thread1.start();
		thread2.start();
  1. 使用join()方法等待两个线程运行结束。这个方法能够抛出InterruptedException异常,所以我们使用代码捕获它:
try {
			thread1.join();
			thread2.join();
		} catch (InterruptedException e) {
			e.printStackTrace();
		}
  1. 在控制台中输出表明程序结束的信息:
System.out.printf("Main: Configuration has been loaded: %s\n", new Date());
  1. 运行程序,查看结果。

工作原理

当运行范例时,需要理解两个线程对象如何开始执行的。首先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()方法与第一种相似,不过它接收参数为毫秒和纳秒两个数值。