多線程學習-day-06ForkJoin實現異步方式遍歷指定文件夾文件

線程基礎、線程之間的共享和協作

(目前會將一些概念簡單描述,一些重點的點會詳細描述)

學習目標:多線程的併發工具類(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的場景有很多,可以留言一起探討。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章