转自 开发者的天空
在有些时候,我们可能需要遍历整个目录树,例如需要寻找所有的.java文件。Java
SE
7提供了很方便的方法来实现这类的功能。
Java SE
7提供的实现这类功能的方法就是FileVisitor接口。FileVisitor接口定义了在遍历中的关键点所需要的行为:当访问文件的时候,在访问
目录前,访问目录后以及出现错误时。对应的这个接口定义了5个方法:
* preVisitDirectory(T) – 在目录被访问前调用。
* preVisitDirectoryFailed(T, IOException) –
当目录不能被访问的时候调用。当访问目录发生异常的时候,异常会被传入该方法,我们可以选择怎样处理这些异常,是直接抛出还是仅仅输出到log文件中等
等。
* postVisitDirectory – Invoked after all the entries in a directory
are visited. If any errors are encountered, the specific exception is
passed to the method.当一个目录中的所有的内容都被访问之后,该方法会被调用。如果有什么错误发生,那么异常会被传入该方法中。
* visitFile
– 当文件被访问的时候该方法会被调用。文件的基本属性会被传入该方法中,或者我们可以使用文件属性包来获取特殊的属性。例如我们可以读取文件的
DosFileAttributeView来判断文件是否被隐藏了。
* visitFileFailed –当文件不能访问的时候,该方法会被调用。异常会被传入该方法,我们可以任意选择怎样处理这个异常。
并不是在所有的情况下我们对这5中情况都需要做特殊的处理,为了简化编程,Java SE
7提供了一个缺省的实现SimpleFileVisitor。该类会遍历目录树
,并抛出所有遇到的异常。我们可以继承该
类,仅仅重写我们需要的实现特殊逻辑的方法。
下面是一个例子,在这个例子中,我们继承了SimplerFileVisitor类。我们要实现的功能是遍历目录树,打印出所有文件/目录的名称和文件大
小。任何碰到的异常都会被输出到控制台中。
import static java.nio.file.FileVisitResult.*;
public static class PrintFiles extends SimpleFileVisitor<Path> {
//Print information about each type of file.
@Override
public FileVisitResult visitFile(Path file, BasicFileAttributes attr) {
if (attr.isSymbolicLink()) {
System.out.format("Symbolic link: %s ", file);
} else if (attr.isRegularFile()) {
System.out.format("Regular file: %s ", file);
} else {
System.out.format("Other: %s ", file);
}
System.out.println("(" + attr.size() + "bytes)");
return CONTINUE;
}
//Print each directory visited.
@Override
public FileVisitResult postVisitDirectory(Path dir, IOException exc) {
System.out.format("Directory: %s%n", dir);
return CONTINUE;
}
//If there is some error accessing the directory, let the user know.
//If you don't override this method and an error occurs, an IOException
//is thrown.
@Override
public FileVisitResult preVisitDirectoryFailed(Path dir, IOException exc) {
System.err.println(exc);
return CONTINUE;
}
//If there is some error accessing the file, let the user know.
//If you don't override this method and an error occurs, an IOException
//is thrown.
@Override
public FileVisitResult visitFileFailed(Path file, IOException exc) {
System.err.println(exc);
return CONTINUE;
}
}
我
们需要的类已经创建好了,下面我们来看看怎样使用这个类。要遍历一个目录有两个方法可以调用,第一个方法仅仅需要传入目录的起点和FileVisitor
的
实例,如下面的代码所示:
Path startingDir = ...;
PrintFiles pf = new PrintFiles();
Files.walkFileTree(startingDir, pf);
第二个方法可以允许我们指定要访问的层数和一组
FileVisitOption作为参数。如果要调用该方法,又想确保整个目录树都被遍历,可以传入Integer.MAX_VALUE作为访问层数的参
数。可以使用的FileVisitOption枚举量有:
* FOLLOW_LINKS – 表明所有的符号链接都要被解析成它们指向的目的地
* DETECT_CYCLES – 检测
循环引用。循环引用是由于符号链接使用不恰当引起
的。
下面的代码演示了怎样调用这个方法:
import static java.nio.file.FileVisitResult.*;
Path startingDir = ...;
EnumSet<FileVisitOption> opts = EnumSet.of(FileVisitOption.FOLLOW_LINKS);
Finder finder = new Finder(pattern);
Files.walkFileTree(startingDir, opts, Integer.MAX_VALUE, finder);
在
遍历目录树的时候是深度优先的。但是在同一级的目录中,我们无法判断哪个会先被访问。如果你的程序要修改文件系统,那么在实现FileVisitor的时
候就需要认真考虑。
例如,如果我们要递归的删除整个目录树,那么我们在删除目录之前要删除其中的所有的文件。我们应该在postVisitDirectory中删除目录本
身。
如果我们是要递归的拷贝整个目录树,我们就需要在preVisitDirectory中创建新的目录,然后在visitFile中拷贝相应的文件。如果要
保持原来目录的属性,那么就在postVisitDirectory中设置新目录的属性。
在遍历目录树的时候,我们还需要决定是否要将符号链接解析为其对应的目的文件/目录。例如我们是要删除目录树,那么就可以不解析符号链接。如果是拷贝目录
树,可能就需要解析。缺省的是不解析的。
在通常情况下只有访问文件时visitFile方法才会被调用。但是,如果我们选择的FOLLOW_LINKS选项并且存在循环链接,那么
visitFile也会被调用,并且该目录会被传入作为参数。因此我们可以使用该特性来判断是否存在循环引用。如:
public FileVisitResult visitFile(Path file, BasicAttributes attrs) {
//It's a circular reference!
if (attrs.isDirectory()) {
...
return CONTINUE;
}
//It's a file.
...
return CONTINUE;
}
可
能我们需要的功能是遍历整个目录树来查找特定的目录。当查找到之后就结束。也有可能我们遍历目录树的时候希望跳过某些目录。我们可以通过这些方法的返回值
来控制程序执行的流程。
FileVisitor的这些方法返回FileVisitResult值,我们通过返回不同的值来控制流程。可选的FileVisitResult值有:
* CONTINUE –表明遍历操作应该继续执行。如preVisitDirectory返回CONTINUE,那么这个目录会被访问。
* TERMINATE – 立即结束遍历操作。
* SKIP_SUBTREE – 当preVisitDirectory方法返回这个值的时候,这个目录及其子目录会被跳过。
* SKIP_SIBLINGS –
当preVisitDirectory方法返回该值的时候,目录不会被访问,postVisitDirectory也不会被调用。其他还没有被访问的兄弟
目录也不会被访问。如果是postVisitDirectory方法返回该值,那么其他没有被访问的兄弟目录不会被访问。
下面的代码中,名为SCCS的目录会被跳过。
import static java.nio.file.FileVisitResult.*;
public FileVisitResult preVisitDirectory(Path dir) {
(if (dir.getName().toString().equals("SCCS")) {
return SKIP_SUBTREE;
}
return CONTINUE;
}
下
面的代码中,一旦特定名称的文件被找到,离开终止遍历操作。
import static java.nio.file.FileVisitResult.*;
//The file we are looking for.
Path lookingFor = ...;
public FileVisitResult visitFile(Path file, BasicFileAttributes attr) {
if (file.getName().equals(lookingFor)) {
System.out.println("Located file: " + file);
return TERMINATE;
}
return CONTINUE;
}
分享到:
相关推荐
Java 实例 - 遍历目录源代码-详细教程.zip
02-Java基础(数组-常见操作-遍历
将Dir.class放在任意需要遍历的文件目录,直接命令窗口运行java Dir生成Dir.txt文件
java设计模式【之】组合模式【源码】【场景:遍历目录树】 * 组合模式 * 将对象组合成树结构,表示 “部分与整体” 的关系 * 要求 部分与整体,具备相同的父类 * * 代码实现 * 输出当前目录下,全部目录层级...
Java 实例 - 遍历指定目录下的所有目录源代码-详细教程.zip
java,利用keySet进行遍历map
同时,书中不仅完整地讲述了由Java SE 8所引入的新特性,例如对包含默认实现代码的接口方法所做的调用,还讲述了为支持类型注解及方法参数注解而对class文件格式所做的扩展,并阐明了class文件中各属性的含义,以及...
遍历一个文件夹下面的所有文件(包括文件夹下的文件夹)
自己总结的。java遍历JSON树
用java程序 遍历某个目录下的所有文件并将文件路径和文件最后修改时间,文件长度写入指定文件中
java 遍历文件目录,区分文件夹及问价类型,方便开发人员对文件操作,用于流方式的拷贝文件
Java遍历目录显示文件列表,读取文件夹中的所有文件,并以列表的形式显示出这些文件,下面来看一下相关的代码: class ShowDirListener implements ActionListener {//取得目录内容的事件处理 public void ...
Java 实例 - 遍历 HashTable 的键值源代码+详细教程.zip
1. 使用Files.list()迭代目录及其子目录文件 Files.list()可以迭代目录及其子目录文件 Files.list(Paths.get(".")) //当前目录 .forEach(System.out::println); 输出: .\filename1.txt .\directory1 .\filename2....
用java实现遍历文件夹中文件并且以树形结构输出
Java遍历目录所有文件,读取指定文件夹下的所有文件名,并列出来,使用java.io.File 类的方法来访问系统中的目录,在窗口上部的文本框中输入c 盘根目录“c:\”,按下窗口下部的按钮“View”,则在窗口中部的文本区...
用递归和堆栈两种方法对树分别进行前中后序的遍历
递归实现遍历目录下子所有目录内所有文件
java全硬盘文件遍历,添加到树中,在面板中显示,没有事件处理
java多叉树的实现:节点集合生成多叉树,单个节点添加到多叉树,深度遍历,广度遍历