列出目錄中的文件
用File類的list()方法可以很容易的列出目錄中的所有文件的文件名。如果想要獲取文件而不止是文件名的話,可以使用它的listFiles()方法。這很簡單,難的是怎么去處理這個返回的列表。我們不再使用傳統的冗長的外部迭代器,而是使用優雅的函數式來實遍歷這個列表。這里我們還得用到JDK的新的CloseableStream接口以及一些相關的高階函數。
下面這段代碼可以列出當前目錄下所有文件的名字。
Files.list(Paths.get("."))
.forEach(System.out::println);
如果想列出別的目錄的話,可以把”.”替換成想要訪問的目錄的完整路徑。
這里先是使用了Paths的get()方法,通過一個字符串創建了一個Path實例。然后通過Files工具類的list()方法獲取到了一個ClosableStream對象,我們可以用它來遍歷目錄下的所有文件。然后我們使用內部迭代器forEach()來打印出文件名。我們先來看下這段代碼的部分輸出結果:列出當前目錄下的文件及子目錄。
./aSampleFiles.txt
./bin
./fpij
...
如果我們只想獲取當前目錄的子目錄,而不要文件的話,可以使用filter()方法:
Files.list(Paths.get("."))
.filter(Files::isDirectory)
.forEach(System.out::println);
ilter()方法將目錄從文件流中篩選出來。我們把Files類的isDirectory方法的引用傳了進去,而不是傳遞一個lambda表達式?;叵胂耭ilter()方法它需要的是一個返回boolean值的Predicate類型,這個方法正好合適。最后我們用一個內部迭代器來打印出目錄的名字。程序將會打印出當前目錄的子目錄。
./bin
./fpij
./output
...
這樣寫簡單多了,跟Java老的寫法相比省了不少代碼。下面我們來看下如何列出匹配某個模式的文件。
列出目錄下指定的文件
Java很早前就提供了一個list()方法的變種,用來篩選文件名。這個版本的list()方法接受一個FilenameFilter類型的參數。這個接口只有一個accept()方法,它接受兩個參數:File dir(代表目錄),以及String name(代表文件名)。accept()方法返回true的話這個文件名就會出現在返回的列表中,返回false則不在。我們來實現一下這個方法。
習慣性的做法是將一個實現了FilenameFilter接口的匿名內部類的實例傳給list()方法。比如說,我們來看下如何用這種方式來返回fpij目錄下的.java文件。
final String[] files =
new File("fpij").list(new java.io.FilenameFilter() {
public boolean accept(final File dir, final String name) {
return name.endsWith(".java");
}
});
System.out.println(files);
這著實得費些工夫寫幾行代碼。這樣的代碼太聒噪了:創建對象,調用函數,定義匿名內部類,在類里面嵌入方法等等。我們不用再忍受這樣的痛苦了,只需傳一個接受兩個參數并返回bollean的lambda表達式進去就好了。Java編譯器會搞定剩下的事。
前面那個例子可以簡單的用一個lambda表達式替換掉匿名內部就好了,但是還有進一步優化的空間。新的DirectoryStream工具可以幫助我們更高效的遍歷大的目錄結構。我們來試下這種方法。這是newDirectoryStream()方法的一個變種,它接受一個額外的過濾器。
Files.newDirectoryStream(
Paths.get("fpij"), path -> path.toString().endsWith(".java"))
.forEach(System.out::println);
這樣我們去掉了匿名內部類并把繁瑣的代碼變得簡潔明了。這兩個版本的輸出結果是一樣的。我們來打印下指定的文件。
這段代碼只會輸出指定目錄下的.java文件,下面是它的部分輸出結果:
fpij/Compare.java
fpij/IterateString.java
fpij/ListDirs.java
...
我們基于文件名來篩選文件,同樣也可以很容易通過文件屬性,比如文件是不是可執行文件,是否可讀,可寫等來進行篩選。這么做的話得需要一個listFiles()方法,它接受一個FileFilter類型的參數。我們仍然使用lambda表達式來實現而不是去創建匿名內部類。現在來看一個列出當前目錄下所有隱藏文件的例子。
final File[] files = new File(".").listFiles(file -> file.isHidden());
如果我們操作的是一個很大的目錄,可以使用DirectoryStream而不是直接調用File上面的方法。
我們傳給listFiles()方法的lambda表達式的簽名和FileFilter接口的accept()方法的簽名是一樣的。這個lambda表達式接受的是一個File實例的參數,在這個例子中參數名是file。如果文件是隱藏屬性的話,剛返回true,否則返回false.
這里其實還可以再精簡下代碼,我們不傳lambda表達式了,傳一個方法引用會讓代碼看起來會更簡潔一些 :
new File(".").listFiles(File::isHidden);
我們先用lambda表達式實現,隨后又使用方法引用將它重構得更加簡潔。如果我們再寫新的代碼的話,當然應該采用這種簡潔的方式。如果可以早點發現這種簡潔的實現,我們當然要優先使用它。有一句話叫做”先讓它能工作,然后再去優化(make it work, then make it better)",先讓代碼能跑起來,等我們理清楚了,才去考慮簡潔性和性能等進行優化。
我們通過一個例子來從目錄中過濾出了指定的文件。下面我們來看下如何去遍歷指定目錄下的子目錄。