使用Files.walkFileTree遍历目录文件

作者:阿拉的梦想 时间:2021-09-27 06:12:40 

java.nio.file.Files.walkFileTree是JDK7新增的静态工具方法。

1.Files.walkFileTree的原理介绍

static Path walkFileTree(Path start, Set<FileVisitOption> options, int maxDepth, FileVisitor<? super Path> visitor) throws IOException;
static Path walkFileTree(Path start, FileVisitor<? super Path> visitor) throws IOException;

参数列表:

  • java.nio.file.Path start 遍历的起始路径

  • Set<java.nio.file.FileVisitOption> options 遍历选项

  • int maxDepth 遍历深度

  • java.nio.file.FileVisitor<? super Path> visitor 遍历过程中的行为控制器

2.遍历行为控制器FileVisitor

接口java.nio.file.FileVisitor包含四个方法,涉及到遍历过程中的几个重要的步骤节点。

一般实际中使用SimpleFileVisitor简化操作。

public interface FileVisitor<T> {

FileVisitResult preVisitDirectory(T dir, BasicFileAttributes attrs)
       throws IOException;

FileVisitResult visitFile(T file, BasicFileAttributes attrs)
       throws IOException;

FileVisitResult visitFileFailed(T file, IOException exc)
       throws IOException;

FileVisitResult postVisitDirectory(T dir, IOException exc)
       throws IOException;
}
  • preVisitDirectory 访问一个目录,在进入之前调用。

  • postVisitDirectory一个目录的所有节点都被访问后调用。遍历时跳过同级目录或有错误发生,Exception会传递给这个方法

  • visitFile 文件被访问时被调用。该文件的文件属性被传递给这个方法

  • visitFileFailed 当文件不能被访问时,此方法被调用。Exception被传递给这个方法。

3.遍历行为结果 FileVisitResult

public enum FileVisitResult {
   CONTINUE,
   TERMINATE,
   SKIP_SUBTREE,
   SKIP_SIBLINGS;
}
  • CONTINUE 继续遍历

  • SKIP_SIBLINGS 继续遍历,但忽略当前节点的所有兄弟节点直接返回上一层继续遍历

  • SKIP_SUBTREE 继续遍历,但是忽略子目录,但是子文件还是会访问

  • TERMINATE 终止遍历

4.查找指定文件

使用java.nio.file.Path提供的startsWith、endsWith等方法,需要特别注意的是:匹配的是路径节点的完整内容,而不是字符串。

例如: /usr/web/bbf.jar

Path path = Paths.get("/usr/web/bbf.jar");
path.endsWith("bbf.jar");  // true
path.endsWith(".jar");     // false

5.使用PathMatcher

@Test
public void visitFile2() throws IOException {
 // 查找java和txt文件
 String glob = "glob:**/*.{java,txt}";
 String path = "D:\\work_java\\bbf\\CORE";

final PathMatcher pathMatcher = FileSystems.getDefault().getPathMatcher(glob);

Files.walkFileTree(Paths.get(path), new SimpleFileVisitor<Path>() {
   @Override
   public FileVisitResult visitFile(Path file, BasicFileAttributes attrs)
       throws IOException {
     if (pathMatcher.matches(file)) {
       System.out.println(file);
     }
     return FileVisitResult.CONTINUE;
   }
 });
}

getPathMatcher方法的参数语法:规则:模式,其中规则支持两种模式glob和regex。

5.1全局规则glob

使用类似于正则表达式但语法更简单的模式,匹配路径的字符串。

  • glob:*.java 匹配以java结尾的文件

  • glob:. 匹配包含&rsquo;.'的文件

  • glob:*.{java,class} 匹配以java或class结尾的文件

  • glob:foo.? 匹配以foo开头且一个字符扩展名的文件

  • glob:/home// 在unix平台上匹配,例如/home/gus/data等

  • glob:/home/** 在unix平台上匹配,例如/home/gus,/home/gus/data

  • glob:c:\\* 在windows平台上匹配,例如c:foo,c:bar,注意字符串转义

5.1.1规则说明

  • * 匹配零个或多个字符与名称组件,不跨越目录

  • ** 匹配零个或多个字符与名称组件,跨越目录(含子目录)

  • ? 匹配一个字符的字符与名称组件

  • \ 转义字符,例如{表示匹配左花括号

  • [] 匹配方括号表达式中的范围,连字符(-)可指定范围。例如[ABC]匹配"A"、&ldquo;B"和"C&rdquo;;[a-z]匹配从"a"到"z";[abce-g]匹配"a"、&ldquo;b&rdquo;、&ldquo;c&rdquo;、&ldquo;e&rdquo;、&ldquo;f&rdquo;、&ldquo;g&rdquo;;

  • [!..]匹配范围之外的字符与名称组件,例如[!a-c]匹配除"a"、&ldquo;b&rdquo;、"c"之外的任意字符

  • {}匹配组中的任意子模式,多个子模式用","分隔,不能嵌套。

5.2正则规则regex

使用java.util.regex.Pattern支持的正则表达式。

5.2.1示例

获取指定扩展名的文件

以下测试用例,目的都是获取指定目录下的.properties和.html文件。

/**
* 递归遍历,字符串判断
*
* @throws IOException IO异常
*/
@Test
public void visitFile1() throws IOException {
 String path = "D:\\work_java\\hty\\HTY_CORE";

Files.walkFileTree(Paths.get(path), new SimpleFileVisitor<Path>() {
   @Override
   public FileVisitResult visitFile(Path file, BasicFileAttributes attrs)
       throws IOException {
     String pathStr = file.toString();
     if (pathStr.endsWith("properties") || pathStr.endsWith("html")) {
       System.out.println(file);
     }
     return FileVisitResult.CONTINUE;
   }
 });
}

/**
* 递归遍历,glob模式
*
* @throws IOException IO异常
*/
@Test
public void visitFile2() throws IOException {
 String glob = "glob:**/*.{properties,html}";
 String path = "D:\\work_java\\hty\\HTY_CORE";

final PathMatcher pathMatcher = FileSystems.getDefault().getPathMatcher(glob);

Files.walkFileTree(Paths.get(path), new SimpleFileVisitor<Path>() {
   @Override
   public FileVisitResult visitFile(Path file, BasicFileAttributes attrs)
       throws IOException {
     if (pathMatcher.matches(file)) {
       System.out.println(file);
     }
     return FileVisitResult.CONTINUE;
   }
 });
}

/**
* 递归遍历,正则模式
*
* @throws IOException IO异常
*/
@Test
public void visitFile3() throws IOException {
 // (?i)忽略大小写,(?:)标记该匹配组不应被捕获
 String reg = "regex:.*\\.(?i)(?:properties|html)";
 String path = "D:\\work_java\\hty\\HTY_CORE";

final PathMatcher pathMatcher = FileSystems.getDefault().getPathMatcher(reg);

Files.walkFileTree(Paths.get(path), new SimpleFileVisitor<Path>() {
   @Override
   public FileVisitResult visitFile(Path file, BasicFileAttributes attrs)
       throws IOException {
     if (pathMatcher.matches(file)) {
       System.out.println(file);
     }
     return FileVisitResult.CONTINUE;
   }
 });
}

6.查找指定文件

/**
* 查找指定文件
*
* @throws IOException IO异常
*/
@Test
public void visitFile() throws IOException {
 String path = "D:\\work_java\\hty\\HTY_CORE\\src";

Files.walkFileTree(Paths.get(path), new SimpleFileVisitor<Path>() {
   @Override
   public FileVisitResult visitFile(Path file, BasicFileAttributes attrs)
       throws IOException {
     // 使用endsWith,必须是路径中的一段,而不是几个字符
     if (file.endsWith("log.java")) {
       System.out.println(file);
       // 找到文件,终止操作
       return FileVisitResult.TERMINATE;
     }
     return FileVisitResult.CONTINUE;
   }
 });
}

7.遍历单层目录

使用DirectoryStream会获取指定目录下的目录和文件。可以使用newDirectoryStream的第二个参数进行筛选,glob语法。

/**
* 遍历单层目录
*
* @throws IOException IO异常
*/
@Test
public void dir() throws IOException {
 Path source = Paths.get("D:\\work_java\\hty\\HTY_CORE\\src\\main\\resources");
 try (DirectoryStream<Path> stream = Files.newDirectoryStream(source, "*.xml")) {
   Iterator<Path> ite = stream.iterator();
   while (ite.hasNext()) {
     Path pp = ite.next();
     System.out.println(pp.getFileName());
   }
 }
}

8.复制文件到新目录

/**
* 递归复制
*
* @throws IOException IO异常
*/
@Test
public void copyAll() throws IOException {
 Path source = Paths.get("D:\\work_java\\hty\\HTY_CORE\\src");
 Path target = Paths.get("D:\\temp\\core");
 // 源文件夹非目录
 if (!Files.isDirectory(source)) {
   throw new IllegalArgumentException("源文件夹错误");
 }
 // 源路径的层级数
 int sourcePart = source.getNameCount();
 Files.walkFileTree(source, new SimpleFileVisitor<Path>() {
   @Override
   public FileVisitResult preVisitDirectory(Path dir, BasicFileAttributes attrs)
       throws IOException {
     // 在目标文件夹中创建dir对应的子文件夹
     Path subDir;
     if (dir.compareTo(source) == 0) {
       subDir = target;
     } else {
       // 获取相对原路径的路径名,然后组合到target上
       subDir = target.resolve(dir.subpath(sourcePart, dir.getNameCount()));
     }
     Files.createDirectories(subDir);
     return FileVisitResult.CONTINUE;
   }

@Override
   public FileVisitResult visitFile(Path file, BasicFileAttributes attrs) throws IOException {
     Files.copy(file, target.resolve(file.subpath(sourcePart, file.getNameCount())),
         StandardCopyOption.REPLACE_EXISTING);
     return FileVisitResult.CONTINUE;
   }
 });
 System.out.println("复制完毕");
}

9.文件和流的复制

/**
* 流复制到文件
*
* @throws IOException IO异常
*/
@Test
public void copy1() throws IOException {
 Path source = Paths.get("D:\\work_java\\hty\\HTY_CORE\\src\\main\\resources\\ehcache.xml");
 Path target = Paths.get("D:\\temp\\");
 if (!Files.exists(target)) {
   Files.createDirectories(target);
 }
 Path targetFile = target.resolve(source.getFileName());
 try (InputStream fs = FileUtils.openInputStream(source.toFile())) {
   Files.copy(fs, targetFile, StandardCopyOption.REPLACE_EXISTING);
 }
}

/**
* 文件复制到流
*
* @throws IOException IO异常
*/
@Test
public void copy2() throws IOException {
 Path source = Paths.get("D:\\work_java\\hty\\HTY_CORE\\src\\main\\resources\\ehcache.xml");
 Path target = Paths.get("D:\\temp\\core");
 Path targetFile = target.resolve(source.getFileName());
 if (!Files.exists(target)) {
   Files.createDirectories(target);
 }
 try (OutputStream fs = FileUtils.openOutputStream(targetFile.toFile());
     OutputStream out = new BufferedOutputStream(fs)) {
   Files.copy(source, out);
 }
}

10.Path与File的转换

/**
* Path与File的转换
*/
@Test
public void testPath() {
File file = new File("D:\\work_java\\hty\\HTY_CORE");
System.out.println(file.toURI());//file:/D:/work_java/hty/HTY_CORE
System.out.println(file.getAbsolutePath());//D:\work_java\hty\HTY_CORE
System.out.println(file.getName());//HTY_CORE

System.out.println("-------");
//File转换为Path
Path path = Paths.get(file.toURI());
System.out.println(path.toUri());//file:///D:/work_java/hty/HTY_CORE
System.out.println(path.toAbsolutePath());//D:\work_java\hty\HTY_CORE
System.out.println(path.getFileName());//HTY_CORE

System.out.println("-------");
//Path转换为File
File f = path.toFile();
System.out.println(f.getAbsolutePath());//D:\work_java\hty\HTY_CORE
}

来源:https://blog.csdn.net/A434534658/article/details/112291879

标签:Files.walkFileTree,遍历,目录文件
0
投稿

猜你喜欢

  • Java通过Fork/Join优化并行计算

    2023-01-27 21:28:36
  • Android Studio与SVN版本控制程序的协作使用指南

    2021-11-08 15:16:08
  • Java算法实现调整数组顺序使奇数位于偶数之前的讲解

    2022-01-23 22:41:19
  • Java 回调callback举例详解

    2023-11-11 16:25:09
  • 安卓11适配攻略抢先看

    2022-05-22 22:39:05
  • nacos注册中心单节点ap架构源码解析(最新推荐)

    2022-09-02 09:31:43
  • Kotlin使用协程实现高效并发程序流程详解

    2021-12-27 20:47:21
  • c#定时器使用示例详解

    2022-01-09 03:31:05
  • SpringMVC RESTFul实战案例访问首页

    2022-03-12 00:21:01
  • Spring存储与读取Bean对象方法

    2021-11-12 03:35:27
  • 一文带你深入了解Java泛型

    2022-02-10 05:38:02
  • C#实战之备忘录的制作详解

    2023-08-13 02:17:21
  • C#执行存储过程并将结果填充到GridView的方法

    2022-08-08 06:25:38
  • C#实现基于Base64的加密解密类实例

    2023-02-09 19:35:57
  • 解析Java的JNI编程中的对象引用与内存泄漏问题

    2023-03-19 20:59:28
  • Spring运行时动态注册bean的方法

    2023-11-25 04:16:58
  • Android禁止横屏竖屏切换的有效方法

    2023-05-05 10:56:16
  • C#获取进程和对进程的操作

    2021-07-14 19:50:35
  • Android启动页出现白屏、黑屏的解决方案

    2022-08-23 21:11:52
  • 详解C#正则表达式Regex常用匹配

    2022-07-08 08:04:45
  • asp之家 软件编程 m.aspxhome.com