对于一个嵌套的目录,获取目录下所有的文件,可以使用以下两种方式:

1. 通过递归获取

File类提供了如下两个方法:

  1. file.list():返回目录下文件和子目录名;(不会递归)
  2. file.listFiles():返回目录下文件和子目录File对象;(不会递归)
/**
     * 得到文件名称
     *
     * @param path 路径
     * @return {@link List}<{@link String}>
     */
    private List<String> getFileNames(String path) {
        File file = new File(path);
        if (!file.exists()) {
            return null;
        }
        List<String> fileNames = new ArrayList<>();
        return getFileNames(file, fileNames);
    }

/**
     * 得到文件名称
     *
     * @param file      文件
     * @param fileNames 文件名
     * @return {@link List}<{@link String}>
     */
    private List<String> getFileNames(File file, List<String> fileNames) {
        File[] files = file.listFiles();
        for (File f : files) {
            if (f.isDirectory()) {
                getFileNames(f, fileNames);
            } else {
                fileNames.add(f.getName());
            }
        }
        return fileNames;
    }

例子中List只是存放了文件夹下所有的文件名,可以根据需求修改List中存放的元素属性。比如可以直接将File对象存入。

2. 使用Files.walk()方法
在jdk8中,可以使用walk方法递归的去查找目录下所有文件。

// 路径
        String path = "D:\\xxx";
		try (Stream<Path> paths = Files.walk(Paths.get(path))){
            List<Path> fileNames = paths
                    .filter(Files::isRegularFile)
                    .collect(Collectors.toList());
        } catch (IOException e) {
            e.printStackTrace();
        }

walk方法会自动递归子目录。
上例使用了Try with Resources模式,它可以确保无论在什么情况下,流都将关闭。
解释:
如果try块和finally块中的方法都抛出异常那么try块中的异常会被抑制(suppress),只会抛出finally中的异常,而把try块的异常完全忽略。而try-with-resources语句能够帮你自动调用资源的close()函数关闭资源不用到finally块。

walk方法解释:

通过遍历以给定起始文件为根的文件树,返回一个用 Path 惰性填充的 Stream。文件树是深度优先遍历的,流中的元素是 Path 对象,就好像通过解析相对路径来获得的一样。此方法的工作方式就像调用它等同于评估表达式: walk(start, Integer.MAX_VALUE, options) 换句话说,它访问文件树的所有级别。返回的流封装了一个或多个 DirectoryStream。如果需要及时处理文件系统资源,则应使用 try-with-resources 构造来确保在流操作完成后调用流的 close 方法。对关闭的流进行操作将导致 illegalStateException。

illegalStateException是无效状态异常。表示当前对客户端的响应已经结束,不能在响应已经结束(或说消亡)后再向客户端(实际上是缓冲区)输出任何内容。
一般来说无效状态异常是因为弄错了调用一个方法的流程,
例如:比如在JSP/Servlet编程中,服务器已经开始把数据发客户端了,却想改动字符集encoding参数,这个就是错误,因为开始复数据到客户端后就不能再修改任何Http header内容,它们已经发出去了,无法再修改了。

maxDepth参数,设置要递归的深度;Files.walk(Paths.get(path),2)
默认不会自动跟随符号链接, 设置options参数FOLLOW_LINKS选项,则遵循符号链接。 Files.walk(Paths.get(path),FileVisitOption.FOLLOW_LINKS)

示例:

String path = "D:\\xxx";
	
	//过滤出目录
	try (Stream<Path> paths = Files.walk(Paths.get(dirName))) {
	    paths.filter(Files::isDirectory)
	            .forEach(System.out::println);
	}
 
	//按后缀名过滤
	try (Stream<Path> paths = Files.walk(Paths.get(dirName), 2)) {
	    paths.map(path -> path.toString()).filter(f -> f.endsWith(".png"))
	            .forEach(System.out::println);
	}

需求:传进来上传的文件名通过","分割,检查文件夹中是否存在,不存在返回缺失的文件名
代码:

/**
     * 校验文件
     *
     * @param data  文件名数据
     * @return {@link String}
     */
    @Override
    public String validationFile(String data) {
        List<String> fileNames = new ArrayList<>();
        try (Stream<Path> paths = Files.walk(Paths.get(path))) {
            fileNames = paths
                    .filter(Files::isRegularFile)
                    .map(file -> file.getFileName().toString())
                    .collect(Collectors.toList());
        } catch (IOException e) {
            e.printStackTrace();
        }
        //传进来的files
        String[] fileNameArr = data.split(",");
        List<String> missingFile = new ArrayList<>();
        for (String s : fileNameArr) {
            if (!fileNames.contains(s)) {
                missingFile.add(s);
            }
        }
        if (missingFile.size() == 0) {
            return "1";
        }
        return String.join(",", missingFile);
    }