欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

Java函数式编程(十一):遍历目录

程序员文章站 2024-02-28 23:41:58
列出目录中的文件 用file类的list()方法可以很容易的列出目录中的所有文件的文件名。如果想要获取文件而不止是文件名的话,可以使用它的listfiles()方法。这很...

列出目录中的文件

用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表达式。回想下filter()方法它需要的是一个返回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)",先让代码能跑起来,等我们理清楚了,才去考虑简洁性和性能等进行优化。

我们通过一个例子来从目录中过滤出了指定的文件。下面我们来看下如何去遍历指定目录下的子目录。