对于一个嵌套的目录,获取目录下所有的文件,可以使用以下两种方式:
1. 通过递归获取
File类提供了如下两个方法:
- file.list():返回目录下文件和子目录名;(不会递归)
- file.listFiles():返回目录下文件和子目录File对象;(不会递归)
private ListgetFileNames(String path) { File file = new File(path); if (!file.exists()) { return null; } List fileNames = new ArrayList<>(); return getFileNames(file, fileNames); } private List getFileNames(File file, List 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 (Streampaths = Files.walk(Paths.get(path))){ List 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 (Streampaths = Files.walk(Paths.get(dirName))) { paths.filter(Files::isDirectory) .forEach(System.out::println); } //按后缀名过滤 try (Stream paths = Files.walk(Paths.get(dirName), 2)) { paths.map(path -> path.toString()).filter(f -> f.endsWith(".png")) .forEach(System.out::println); }
需求:传进来上传的文件名通过","分割,检查文件夹中是否存在,不存在返回缺失的文件名
代码:
@Override public String validationFile(String data) { ListfileNames = new ArrayList<>(); try (Stream 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 missingFile = new ArrayList<>(); for (String s : fileNameArr) { if (!fileNames.contains(s)) { missingFile.add(s); } } if (missingFile.size() == 0) { return "1"; } return String.join(",", missingFile); }