Java如何使用Files.walkFileTree遍历目录文件呢?

书欣 Java经验 发布时间:2022-10-15 13:24:01 阅读数:5469 1
下文笔者讲述使用Files.walkFileTree这个静态工具方法即可遍历指定路径,如下所示

Files.walkFileTree简介

java.nio.file.Files.walkFileTree:
     是JDK7新增的静态工具方法
walkFileTree()方法的功能:
     用于递归遍历目录树的方法
     方法参数接收一个Path对象和一个FileVisitor对象
     Path对象:指需要遍历的目录
     FileVisitor遍历的参数选项

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:
      遍历过程中的行为控制器

遍历行为控制器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被传递给这个方法。

遍历返回对象FileVisitResult

public enum FileVisitResult {
    CONTINUE,
    TERMINATE,
    SKIP_SUBTREE,
    SKIP_SIBLINGS;
} 
CONTINUE 继续遍历
SKIP_SIBLINGS 继续遍历,但忽略当前节点的所有兄弟节点直接返回上一层继续遍历
SKIP_SUBTREE 继续遍历,但是忽略子目录,但是子文件还是会访问
TERMINATE 终止遍历
例:

查找指定文件

使用java.nio.file.Path提供的startsWith、endsWith等方法
注意事项:
    匹配路径参数是一个Path对象
	不是一个字符串的完整
例:
/opt/test123.jar
  
Path path = Paths.get("/opt/test123.jar");
path.endsWith("test123.jar");  // true
path.endsWith(".jar");     // false
例:
使用PathMatcher查找java和txt文件
@Test
public void visitFile2() throws IOException {
  // 查找java和txt文件
  String glob = "glob:**/*.{java,txt}";
  String path = "D:\\test";

  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。

全局规则glob

使用类似于正则表达式但语法更简单的模式
匹配路径的字符串
glob:*.java 匹配以java结尾的文件
glob:. 匹配包含’.'的文件
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"、“B"和"C”;[a-z]匹配从"a"到"z";[abce-g]匹配"a"、“b”、“c”、“e”、“f”、“g”;
[!..]匹配范围之外的字符与名称组件,例如[!a-c]匹配除"a"、“b”、"c"之外的任意字符
{}匹配组中的任意子模式,多个子模式用","分隔,不能嵌套。
5.2正则规则regex
使用java.util.regex.Pattern支持的正则表达式。

获取指定扩展名的文件

例:
获取指定目录下的.properties和.html文件
/**
 * 递归遍历,字符串判断
 *
 * @throws IOException IO异常
 */
@Test
public void visitFile1() throws IOException {
  String path = "D:\\test";

  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:\\test";

  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:\\test";

  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;
    }
  });
}

查找指定文件

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

  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;
    }
  });
}

遍历单层目录

使用DirectoryStream会获取指定目录下的目录和文件
可使用newDirectoryStream的第二个参数进行筛选
glob语法
/**
 * 遍历单层目录
 *
 * @throws IOException IO异常
 */
@Test
public void dir() throws IOException {
  Path source = Paths.get("D:\\test");
  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());
    }
  }
}

复制文件到新目录

/**
 * 递归复制
 *
 * @throws IOException IO异常
 */
@Test
public void copyAll() throws IOException {
  Path source = Paths.get("D:\\test1");
  Path target = Paths.get("D:\\test2");
  // 源文件夹非目录
  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("复制完毕");
}

文件和流的复制

/**
 * 流复制到文件
 *
 * @throws IOException IO异常
 */
@Test
public void copy1() throws IOException {
  Path source = Paths.get("D:\\test1\test.txt");
  Path target = Paths.get("D:\\test2\\");
  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:\\test1\\test.txt");
  Path target = Paths.get("D:\\test2");
  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);
  }
}

Path与File的转换

/**
 * Path与File的转换
 */
@Test
public void testPath() {
	File file = new File("D:\\test1\\test_name");
	System.out.println(file.toURI());//file:/D:/test
	System.out.println(file.getAbsolutePath());//D:\test\test_name
	System.out.println(file.getName());//test_name
	
	System.out.println("-------");
	//File转换为Path
	Path path = Paths.get(file.toURI());
	System.out.println(path.toUri());//file:///D:/test/test_name
	System.out.println(path.toAbsolutePath());//D:\test\test_name
	System.out.println(path.getFileName());//test_name
	
	System.out.println("-------");
	//Path转换为File
	File f = path.toFile();
	System.out.println(f.getAbsolutePath());//D:\test\test_name
}
版权声明

本文仅代表作者观点,不代表本站立场。
本文系作者授权发表,未经许可,不得转载。

本文链接: https://www.Java265.com/JavaJingYan/202210/16658114904644.html

最近发表

热门文章

好文推荐

Java265.com

https://www.java265.com

站长统计|粤ICP备14097017号-3

Powered By Java265.com信息维护小组

使用手机扫描二维码

关注我们看更多资讯

java爱好者