Java 8 - 根据谓词递归删除文件夹

标签 java java-8 functional-programming java-stream

要求是遍历一系列文件夹,以根据文件夹名称模式识别和删除子文件夹(任何级别)及其内容。我有使用 Java 7 的访问者类的工作代码,但现在尝试使其与 Java 8 流一起工作。

我想出了一些工作代码(见下文),但我的印象是它应该存在一种更好、更简单的方法来完成同样的事情,而无需重复自己。

  public void delete(List<Path> folders) {
     folders
     .stream()
     .filter(Files::isDirectory)
     .forEach(this::evaluateIfTrash);
  }

  private void evaluateIfTrash(Path folder) {
    try (Stream<Path> dir = Files.walk(folder, FileVisitOption.FOLLOW_LINKS)) {
      dir.filter(Files::isDirectory)
          .filter(TrashPredicate::isTrashFolder)
          .forEach(this::recursivelyDelete);
    } catch (IOException e) {
      log.error("Failed to evaluate {}", folder, e);
    }
  }

  private void recursivelyDelete(Path folder) {
    try (Stream<Path> dir = Files.walk(folder, FileVisitOption.FOLLOW_LINKS)) {
      dir.sorted(Comparator.reverseOrder())
          .map(Path::toFile)
          .filter(File::exists)
          .forEach(File::delete);
    } catch (IOException e) {
      log.error("Failed to delete {}", folder, e);
    }
  }

在阅读了一些文档和几个答案后,我尝试使用嵌套流来消除一些重复:

 private void delete(List<Path> folders) {
    folders
    .stream()
    .filter(Files::isDirectory)
    .flatMap(
        f -> {
          try {
            return Files.walk(f, FileVisitOption.FOLLOW_LINKS);
          } catch (IOException e) {
            return Stream.empty();
          }
        })
    .filter(TrashPredicate::isTrashFolder)
    .forEach(this::recursivelyDelete);
  }

private void recursivelyDelete(Path folder) {
    try (Stream<Path> dir = Files.walk(folder, FileVisitOption.FOLLOW_LINKS)) {
      dir.sorted(Comparator.reverseOrder())
          .map(Path::toFile)
          .filter(File::exists)
          .forEach(File::delete);
    } catch (IOException e) {
      log.error("Failed to delete {}", folder, e);
    }
  }

这消除了其中一种方法(即使可读性不太好),但我真正不喜欢的是没有关闭 flatMap 操作返回的内部流。当我尝试这样做时(通过使用 try-with-resources 构造并在 try 内移动过滤器和 forEach 操作),我收到一条错误消息,指出流已关闭且不可用于进一步操作。

过去两天我一直在尝试让这项工作成功,但没有成功,所以问题又来了,是否有更好(更干净)的方法使用 Java 8 来完成这项工作?

最佳答案

Files.walk 已经以深度优先递归方式传输根路径。来自apache commons的FileUtils有一个deleteDirectory,它可以递归地删除目录,使代码也非常干净。

所以像下面这样的东西应该有效

        Files.walk(rootPath)
            .filter(Files::isDirectory).
            .filter(TrashPredicate::isTrashFolder)
            .forEach(FileUtils::deleteDirectory);

关于Java 8 - 根据谓词递归删除文件夹,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49373994/

相关文章:

recursion - 降低 Racket 的功能?

java - Ehcache 设置为永恒但无论如何都会忘记元素?

java - 无法将外部 jar 添加到应用程序 jar 文件中

java - Firebase 数据插入无法正常工作,即使它说已连接到应用程序

java - 在Android Studio中将Java变量插入xml

java - 使用 Java8 流 API 从 token 分隔输入流中解码并行消息

java - 如何从泛型列表转换为 Java 8 中的对象列表?

java - 如何将这个经典的 Java 代码重写为 Java Stream API 代码?

functional-programming - OCaml 可选参数

javascript - 函数式 JS 设置对象的所有属性