Java 7 walkFileTree在目录上调用visitFile

Mat*_*son 2 java-7 nio2

说我有以下目录结构

/root/dir
/root/dir/file1.txt
/root/dir/subdir
/root/dir/subdir/file2.txt
Run Code Online (Sandbox Code Playgroud)

假设我将使用以下访客:

class MyFileVisitor extends SimpleFileVisitor<Path> {

  @Override
  public FileVisitResult visitFile(final Path file, final BasicFileAttributes attrs)
  throws IOException {
    if(Files.isDirectory(file)) {
      throw new IllegalStateException("WAT!? Visiting directory: "+file.toAbsolutePath().toString());
    }
    System.out.println("Visiting file: "+file.toAbsolutePath().toString());
    return super.visitFile(file, attrs);
  }
Run Code Online (Sandbox Code Playgroud)

如果我们使用walkFileTree的简单重载:

Files.walkFileTree(Paths.get("/root/dir"), new MyFileVisitor());  
Run Code Online (Sandbox Code Playgroud)

一切都按计划进行,我们看到以下输出:

Visiting file: /root/dir/file1.txt
Visiting file: /root/dir/subdir/file2.txt
Run Code Online (Sandbox Code Playgroud)

但是,当我们尝试设置最大深度时,事情开始崩溃:

Files.walkFileTree(Paths.get("/root/dir"), EnumSet.noneOf(FileVisitOption.class), 1, new MyFileVisitor());
Run Code Online (Sandbox Code Playgroud)

输出:

Visiting file: /root/dir/file1.txt

java.lang.IllegalStateException: WAT!? Visting directory: /root/dir/subdir
Run Code Online (Sandbox Code Playgroud)

我很确定这是一个错误,但是我想先问一下社区:我缺少什么了吗,这实际上是预期的行为吗?感谢您的确认!

细节:

java -version
java version "1.7.0_45"
Java(TM) SE Runtime Environment (build 1.7.0_45-b18)
Java HotSpot(TM) 64-Bit Server VM (build 24.45-b08, mixed mode)
Run Code Online (Sandbox Code Playgroud)

Fra*_*ner 5

我用了一些代码,并增加了几行:

/*
 * (non-Javadoc)
 * 
 * @see java.nio.file.SimpleFileVisitor#preVisitDirectory(java.lang.Object,
 * java.nio.file.attribute.BasicFileAttributes)
 */
@Override
public FileVisitResult preVisitDirectory(Path dir, BasicFileAttributes attrs)
        throws IOException {
    System.out.println(Thread.currentThread().getStackTrace()[1] + " "
            + dir);
    return super.preVisitDirectory(dir, attrs);
}

/*
 * (non-Javadoc)
 * 
 * @see java.nio.file.SimpleFileVisitor#postVisitDirectory(java.lang.Object,
 * java.io.IOException)
 */
@Override
public FileVisitResult postVisitDirectory(Path dir, IOException exc)
        throws IOException {
    System.out.println(Thread.currentThread().getStackTrace()[1] + " "
            + dir);
    return super.postVisitDirectory(dir, exc);
}

/*
 * (non-Javadoc)
 * 
 * @see java.nio.file.SimpleFileVisitor#visitFileFailed(java.lang.Object,
 * java.io.IOException)
 */
@Override
public FileVisitResult visitFileFailed(Path file, IOException exc)
        throws IOException {
    System.out.println(Thread.currentThread().getStackTrace()[1] + " "
            + file);
    return super.visitFileFailed(file, exc);
}
Run Code Online (Sandbox Code Playgroud)

不用担心,看看发生了什么。

您所说的错误发生在这里java.nio.file.FileTreeWalker:134

    // at maximum depth or file is not a directory
    if (depth >= maxDepth || !attrs.isDirectory()) {
        return visitor.visitFile(file, attrs);
    }
Run Code Online (Sandbox Code Playgroud)

最后,我对此有何看法:

1 .: subdir是深度1的一部分
。2 .:如果没有visitFilefor最深的目录,您甚至不会意识到它们在那里。

所以我认为这是正常行为

顺便说一句,您可以在覆盖以下内容时使用attrs:

@Override
public FileVisitResult visitFile(final Path file,
        final BasicFileAttributes attrs) throws IOException {
    if (attrs.isDirectory()) {
        throw new IllegalStateException("WAT!? Visiting directory: "
                + file.toAbsolutePath().toString());
    }
    System.out
            .println("Visiting file: " + file.toAbsolutePath().toString());
    return super.visitFile(file, attrs);
}
Run Code Online (Sandbox Code Playgroud)