線程基礎、線程之間的共享和協作
(目前會將一些概念簡單描述,一些重點的點會詳細描述)
學習目標:多線程的併發工具類(2)
利用ForkJoin來寫一個異步方式遍歷指定文件夾下所有文件(或指定文件)程序
直接看代碼吧,註釋也比較詳細了:
/**
* 異步遍歷指定盤的所有文件
*
* @author Administrator
*
*/
public class FindDirFiles extends RecursiveAction {
private static final long serialVersionUID = 7698200241798928022L;
private File path; // 當前要搜索的目錄
public FindDirFiles(File path) {
this.path = path;
}
@Override
protected void compute() {
// 定義一個文件目錄集合
List<FindDirFiles> subTasks = new ArrayList<>();
// 根據當前要搜索目錄的,找到所有的文件
File[] files = path.listFiles();
// 判斷是否爲空,不爲空則繼續往下搜索
if (null != files) {
// 不爲空,則進行循環判斷
for (File file : files) {
// 判斷是否是目錄
if (file.isDirectory()) {
// 判斷是目錄,則保存到一個文件集合中
subTasks.add(new FindDirFiles(file));
} else {
// 判斷不是目錄,則是文件,則輸出文件名稱,這裏輸出所有的文件,我這裏輸出太多,因此先輸出指定文件
// System.out.println(file.getAbsolutePath());
// 如果遇到txt文件,則輸出
if (file.getAbsolutePath().endsWith("xmind")) {
System.out.println("文件是:" + file.getAbsolutePath());
}
}
}
// 判斷集合是否爲空
if (null != subTasks && subTasks.size() > 0) {
// 進行循環,把所有任務都invokeAll(),
// 因爲public <T> List<Future<T>> invokeAll(Collection<? extends
// Callable<T>> tasks)
// 可看出,invoke本身也是返回一個集合的結果
for (FindDirFiles subTask : invokeAll(subTasks)) {
// 等待子任務完成
subTask.join();
}
}
}
}
public static void main(String[] args) {
try {
System.out.println("線程開始.......");
// 定義ForkJoinPool對象池實例調度總任務
ForkJoinPool forkJoinPool = new ForkJoinPool();
// 初始化FindDirFiles對象
FindDirFiles findDirFiles = new FindDirFiles(new File("E:/"));
// 異步執行,如果是同步執行則用invoke
forkJoinPool.execute(findDirFiles);
// 證明是異步執行
System.out.println("這裏我們要做一件從0加到100的求和");
// 休眠1秒
Thread.sleep(1000);
// 循環計算0-100的求和運算
int sum = 0;
for (int i = 0; i < 101; i++) {
sum += i;
}
System.out.println("求和結果算出來了,sun = " + sum);
// FindDirFiles對象要進行阻塞
findDirFiles.join();
System.out.println("線程結束......");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
控制檯輸出結果:
線程開始.......
這裏我們要做一件從0加到100的求和
文件是:E:\辦公文件\2014下半年文件\2014年學校+辦公文件\實習工作\2014-04-21\電子期刊軟件\數據標準專刊\歷史\項目專欄規劃.xmind
文件是:E:\辦公文件\2014下半年文件\2014年學校+辦公文件\實習工作\2014-04-28\電子期刊軟件\數據標準專刊\歷史\項目專欄規劃.xmind
求和結果算出來了,sun = 5050
在這裏我們可以看到,輸出的結果卻是是異步的。
一、ForkJoinPool異步和同步的區別:
1,調用execute()方法時,爲異步執行
2,調用invoke()方法時,爲同步執行
二、invokeAll()方法
1,invokeAll()方法, 我們查看源代碼的實現可以看出,返回一個集合形式的結果,因此可以在for循環的泛型裏面直接用invokeAll(集合對象)方法
2、調用了invokeAll()方法後,將所有循環內的子方法都join()起來,等待子任務的完成
因此用Fork/Join的場景有很多,可以留言一起探討。