Java7遍历文件夹

本文介绍使用Java7的新特性遍历文件夹的方法。通过Files.walkFileTree方法配合FileVisitor接口,可以实现对指定路径下所有文件及子目录的遍历,并在访问文件夹和文件时触发相应回调。

Java遍历文件夹的方法有多种,之前我写了一种遍历文件夹的方法点击,使用File类的方法递归遍历,这次是使用Java7提供的新的方法遍历文件夹。

File提供两个方法遍历文件夹,

上面两个方法都需要FileVisitor参数,FileVisitor代表一个文件访问器,walkFileTree()方法会自动遍历start路径下的所有文件和子目录,遍历文件和子目录都会触发FileVisitor中相应的方法。这四个方法在下面的代码中出现。

FileVisitor中的四个方法会返回一个FileVisitorResult,它代是一个枚举类,代表访问之后的行为。

FileVisitor定义了如下几种行为:

CONTINUE:代表访问之后的继续行为

SKIP_SIBLINGS:代表继续访问,但不访问该文件或目录的兄弟文件或目录

SKIP_SUBTREE:继续访问,但不访问该目录或文件的子目录

TERMINATE:终止访问

import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.CharBuffer;
import java.nio.charset.Charset;
import java.nio.charset.CharsetDecoder;
import java.nio.charset.CharsetEncoder;
import java.nio.file.FileVisitResult;
import java.nio.file.FileVisitor;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.nio.file.attribute.BasicFileAttributes;
public class Test {
	public static void main(String[] args) throws Exception
	{
		
		Files.walkFileTree(Paths.get("D:","linux"), new FileVisitor(){

			@Override
			public FileVisitResult preVisitDirectory(Object dir,
					BasicFileAttributes attrs) throws IOException {
				//访问文件夹之前调用
				System.out.println(dir);
				return FileVisitResult.CONTINUE;
			}

			@Override
			public FileVisitResult visitFile(Object file,
					BasicFileAttributes attrs) throws IOException {
				// 访问文件调用
				System.out.println(file);
				return FileVisitResult.CONTINUE;
			}

			@Override
			public FileVisitResult visitFileFailed(Object file, IOException exc)
					throws IOException {
				// 访问文件失败时调用
				return FileVisitResult.CONTINUE;
			}

			@Override
			public FileVisitResult postVisitDirectory(Object dir,
					IOException exc) throws IOException {
				// 访问文件夹之后调用
				return FileVisitResult.CONTINUE;
			}
			
		});
	}
}


这样就可以遍历一个路径下的所有文件和文件夹。

<think> 我们有两个站内引用,都是关于Java递归遍历文件夹的。用户现在明确要求使用Java语言实现递归遍历文件夹方法。 引用[1]和引用[2]都是Java代码,但两者在处理上略有不同: 引用[1]:只是简单地遍历并打印文件(或文件夹)的名字。 引用[2]:在遍历文件时,会读取文件内容并匹配正则表达式,然后打印匹配到的文件路径。 根据用户需求,他想要的是“递归遍历文件夹”,所以我们主要参考引用[1]的基本结构,因为引用[2]还包含了文件内容读取(这可能是用户不需要的)。 但是,我们也可以考虑提供两种方式:一种是简单遍历(仅列出文件),另一种是带过滤条件的遍历(比如按文件名过滤,或者按内容过滤)。不过用户没有要求过滤,所以先给出基本遍历。 设计思路: 1. 定义一个递归方法,参数为File对象(表示当前要遍历文件夹文件)。 2. 首先判断File对象是否为空或者不存在,如果不存在则返回(或打印信息)。 3. 如果当前File文件夹,则获取其下的所有文件和子文件夹(使用listFiles()方法),然后对每个子项递归调用该方法。 4. 如果当前File文件,则直接处理(如打印路径)。 注意:为了避免死循环,我们不需要特殊处理"."和"..",因为Java的listFiles()方法已经不会返回这两个特殊目录(与C++不同)。 另外,用户可能希望得到文件列表,我们可以将结果收集到一个List中返回,也可以直接打印。 下面提供两种实现方式: 方式一:仅打印文件路径(不返回列表) 方式二:收集所有文件路径并返回列表 同时,为了代码的健壮性,需要处理可能的空指针(比如当访问一个没有权限的文件夹时,listFiles()可能返回null)。 步骤: 1. 判断文件对象是否为null,如果为null则返回。 2. 判断文件是否存在,如果不存在则返回。 3. 如果是文件夹,则获取其下的所有文件和子文件夹(注意:这里要检查listFiles()返回的是否为null,避免空指针异常)。 4. 对每个子文件/文件夹递归调用。 下面分别实现这两种方式。 方式一:直接打印文件路径(不返回列表) 方式二:返回文件列表 同时,我们可以考虑提供一个入口方法,允许用户传入一个路径字符串,然后在内部转换为File对象。 另外,用户可能希望遍历时忽略隐藏文件或系统文件,但JavaFile类并不直接提供判断隐藏文件方法,我们可以通过文件名来判断(例如以'.'开头的文件在Unix-like系统上是隐藏文件),但这并不是跨平台的。如果需要,我们可以使用Files.isHidden()方法(NIO包),但为了简单,我们先不考虑。 代码实现如下: </think> ### Java 递归遍历文件夹的实现方法Java 中递归遍历文件夹需要结合 `java.io.File` 类,通过判断文件类型(目录/文件)实现递归逻辑。以下是两种常用实现方式,基于您提供的站内引用[^1][^2]优化而来: --- #### 方法 1:基础遍历(仅列出文件) 适用于只需获取文件路径的场景: ```java import java.io.File; public class FileTraversal { public static void main(String[] args) { File root = new File("C:/HWKJ/file"); // 替换为实际路径 traverseFiles(root); } public static void traverseFiles(File file) { if (file == null || !file.exists()) { System.out.println("路径不存在"); return; } if (file.isDirectory()) { // 递归步骤:处理文件夹 File[] files = file.listFiles(); if (files != null) { // 避免空指针异常 for (File f : files) { traverseFiles(f); // 递归调用 } } } else { // 基本情况:处理文件 System.out.println("文件路径: " + file.getAbsolutePath()); } } } ``` **关键点说明**: 1. **递归终止条件**:当 `file` 是普通文件时直接输出路径 2. **递归过程**: - 通过 `file.isDirectory()` 判断是否为文件夹 - 使用 `listFiles()` 获取子项(返回 `File[]` 数组) - 遍历数组并递归调用自身 3. **安全防护**: - 检查 `file.exists()` 避免无效路径 - 验证 `files != null` 防止无权限访问的文件夹导致崩溃 --- #### 方法 2:带内容过滤的遍历 适用于需要搜索特定文件内容的场景(基于引用[^2]优化): ```java import java.io.*; import java.util.*; public class FileSearch { public static void main(String[] args) throws IOException { File root = new File("E:\\demo"); // 替换为实际路径 List<File> result = searchFiles(root, ".*te.*"); // 正则表达式过滤 result.forEach(f -> System.out.println("匹配文件: " + f.getAbsolutePath())); } public static List<File> searchFiles(File file, String regex) throws IOException { List<File> result = new ArrayList<>(); if (file == null || !file.exists()) return result; if (file.isDirectory()) { File[] files = file.listFiles(); if (files != null) { for (File f : files) { result.addAll(searchFiles(f, regex)); // 递归收集结果 } } } else { try (BufferedReader br = new BufferedReader(new FileReader(file))) { String line; while ((line = br.readLine()) != null) { if (line.matches(regex)) { // 正则匹配内容 result.add(file); break; // 匹配成功即跳出 } } } } return result; } } ``` **增强功能**: - 使用正则表达式 `line.matches(regex)` 过滤文件内容 - 通过 `List<File>` 收集结果而非直接打印 - 资源自动管理:`try-with-resources` 确保 `BufferedReader` 关闭 - 匹配后立即 `break` 提升搜索效率 --- ### 关键注意事项 1. **空指针防护**: - 始终检查 `file.listFiles()` 是否为 `null`(无权限目录会返回 `null`) - 验证 `file.exists()` 避免无效路径 2. **性能优化**: - 超深目录可能引发 `StackOverflowError`,可改用栈迭代(见下文扩展问题) - 内容搜索时优先过滤文件名(如 `f.getName().contains("te")`)减少 IO 开销 3. **跨平台兼容**: - 路径分隔符使用 `File.separator` 替代硬编码 `\` 或 `/` - 隐藏文件处理:通过 `file.isHidden()` 跳过(需额外判断) > 递归遍历的时间复杂度为 $O(n)$($n$ 为文件和目录总数),空间复杂度 $O(d)$($d$ 为目录深度)[^1][^2]。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值