Java并發(fā)編程示例(六):等待線程執(zhí)行終止
在某些場景下,我們必須等待線程執(zhí)行完成才能進(jìn)行下一步工作。例如,某些程序在開始執(zhí)行之前,需要先初始化一些資源。這時,我們可以啟動一個線程專門來做初始化任務(wù),等到線程任務(wù)完成后,再去執(zhí)行其他部分。
為此,Thread類為我們提供了join()方法。當(dāng)我們使用線程對象調(diào)用此方法時,正在掉調(diào)用的線程對象將被推遲到被調(diào)用對象執(zhí)行完成后再開始執(zhí)行。
在本節(jié),示例程序演示等待初始化方法完成后,再去執(zhí)行其他任務(wù)。
知其然
按照下面所示步驟,完成示例程序。
1.創(chuàng)建一個名為DataSourcesLoader的類,并且實(shí)現(xiàn)Runnable接口。代碼如下:
public class DataSourcesLoader implements Runnable {
2.實(shí)現(xiàn)run()方法,向控制臺打印出一條信息以說明開始執(zhí)行,然后睡眠4秒鐘,再向控制臺打印一條信息來說明線程執(zhí)行結(jié)束。代碼如下:
@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());
}
3.創(chuàng)建一個名為NetworkConnectionsLoader的類,并且實(shí)現(xiàn)Runnable接口。 實(shí)現(xiàn)run()方法,該方法代碼與DataSourcesLoader類的run()方法一樣,只是這個睡眠6秒鐘。
4.實(shí)現(xiàn)示例的主類,并且實(shí)現(xiàn)main()方法。代碼如下:
public class Main {
public static void main(String[] args) {
5.創(chuàng)建一個DataSourcesLoader對象,以及一個啟動它執(zhí)行的Thread對象。代碼如下:
DataSourcesLoader dsLoader = new DataSourcesLoader();
Thread thread1 = new Thread(dsLoader, "DataSourcesLoader");
6.創(chuàng)建一個NetworkConnectionsLoader對象,以及一個啟動它執(zhí)行的Thread對象。代碼如下:
NetworkConnectionsLoader ncLoader = new NetworkConnectionsLoader();
Thread thread2 = new Thread(ncLoader, "NetworkConnectionsLoader");
7.調(diào)用兩個Thread對象的start()方法。代碼如下:
thread1.start();
thread2.start();
8.調(diào)用join()方法,來等待兩個線程完成其任務(wù)。這個方法會拋出InterruptedException異常,所以要捕獲該異常。代碼如下:
try {
thread1.join();
thread2.join();
} catch (InterruptedException e) {
e.printStackTrace();
}
9.向控制臺打印一句話,表明程序執(zhí)行結(jié)束。代碼如下:
System.out.printf("Main: Configuration has been loaded: %s\n",
new Date());
10.運(yùn)行程序,查看執(zhí)行效果。
知其所以然
當(dāng)運(yùn)行這個示例程序時,我們可以看到兩個線程啟動了它們的執(zhí)行。首先,是DataSourcesLoader完成了它的執(zhí)行;然后,是NetworkConnectionsLoader完成了它的執(zhí)行。這時,主線程繼續(xù)它的執(zhí)行,然后向控制臺打印出終止信息。
永無止境
Java提供了另外兩種重載的join()方法:
join(long milliseconds)
join(long milliseconds, long nanos)
第一種方式,不會直到被調(diào)用完成任務(wù),而是等待參數(shù)指定的時間后就開始執(zhí)行;例如,如果thread1調(diào)用該方法,thread1.join(1000),當(dāng)thread1線程滿足如下其中之一的條件就會繼續(xù)執(zhí)行:
1.thread2完成它的執(zhí)行;
2.1000毫秒過后;
當(dāng)這兩個條件中的其中之一為真時,join()方法就會返回,開始繼續(xù)執(zhí)行原來的任務(wù)。
第二種方式的方法和第一種很類似,只是多了一個納秒級的時間參數(shù)。
拿來主義
本文是從 《Java 7 Concurrency Cookbook》 (D瓜哥竊譯為 《Java7并發(fā)示例集》 )翻譯而來,僅作為學(xué)習(xí)資料使用。沒有授權(quán),不得用于任何商業(yè)行為。
小有所成
本節(jié)所用示例代碼的完整版。
DataSourcesLoader類的完整代碼
package com.diguage.books.concurrencycookbook.chapter1.recipe6;
import java.util.Date;
import java.util.concurrent.TimeUnit;
/**
* Date: 2013-09-19
* Time: 09:15
*/
public class DataSourcesLoader implements Runnable {
@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類的完整代碼
package com.diguage.books.concurrencycookbook.chapter1.recipe6;
import java.util.Date;
import java.util.concurrent.TimeUnit;
/**
* Date: 2013-09-19
* Time: 09:21
*/
public class NetworkConnectionsLoader implements Runnable {
@Override
public void run() {
System.out.printf("Beginning data sources loading: %s\n",
new Date());
try {
TimeUnit.SECONDS.sleep(6);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.printf("Data sources loading has finished: %s\n",
new Date());
}
}
Main類的完整代碼
package com.diguage.books.concurrencycookbook.chapter1.recipe6;
import java.util.Date;
/**
* Date: 2013-09-19
* Time: 09:25
*/
public class Main {
public static void main(String[] args) {
DataSourcesLoader dsLoader = new DataSourcesLoader();
Thread thread1 = new Thread(dsLoader, "DataSourcesLoader");
NetworkConnectionsLoader ncLoader = new NetworkConnectionsLoader();
Thread thread2 = new Thread(ncLoader, "NetworkConnectionsLoader");
thread1.start();
thread2.start();
try {
thread1.join();
thread2.join();
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.printf("Main: Configuration has been loaded: %s\n",
new Date());
}
}
相關(guān)文章
SpringCloud Bus消息總線的實(shí)現(xiàn)
消息總線是一種通信工具,可以在機(jī)器之間互相傳輸消息、文件等,這篇文章主要介紹了SpringCloud Bus消息總線的實(shí)現(xiàn),Spring cloud bus 通過輕量消息代理連接各個分布的節(jié)點(diǎn),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-05-05Mybatis中xml的動態(tài)sql實(shí)現(xiàn)示例
本文主要介紹了Mybatis中xml的動態(tài)sql實(shí)現(xiàn)示例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-06-06Java的Hibernate框架中用于操作數(shù)據(jù)庫的HQL語句講解
這篇文章主要介紹了Java的Hibernate框架中用于操作數(shù)據(jù)庫的HQL語句講解,Hibernate是Java的SSH三大web開發(fā)框架之一,需要的朋友可以參考下2016-01-01Spring Boot與Kotlin 整合全文搜索引擎Elasticsearch的示例代碼
本篇文章主要介紹了Spring Boot與Kotlin 整合全文搜索引擎Elasticsearch的示例代碼,具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-01-01oracle數(shù)據(jù)庫導(dǎo)入TXT文件方法介紹
這篇文章主要介紹了oracle數(shù)據(jù)庫導(dǎo)入TXT文件方法介紹,文中向大家展示了具體代碼示例,需要的朋友可以參考下。2017-09-09JVM內(nèi)存區(qū)域劃分相關(guān)原理詳解
這篇文章主要介紹了JVM內(nèi)存區(qū)域劃分相關(guān)原理詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-10-10