Java NIO.2 的 Path 和 Files 类取代 java.io.File,Path 抽象路径结构且不可变,Files 提供静态 I/O 方法;支持符号链接、原子操作、批量属性访问及更清晰异常体系。
Java NIO.2 的 Path 和 Files 类是现代文件操作的核心,取代了传统 的许多缺陷,支持符号链接、原子操作、批量属性访问和更清晰的异常体系。
java.io.File
Path 不是文件本身,而是对文件系统中位置的不可变引用。它不检查路径是否存在,只负责表示路径结构和提供解析能力。
Paths.get() 快速创建:Path p = Paths.get("data", "logs", "app.log");\,Unix 用 /)Path p = Paths.get(MyClass.class.getResource("/config.json").toURI());
p.getParent() → 上级目录;p.getFileName() → 最末段(含扩展名);p.resolve("backup.txt") → 拼接子路径(安全,不依赖当前目录);p.relativize(other) → 计算相对路径,例如 Paths.get("a/b").relativize(Paths.get("a/c/d")) 返回 ../c/d
Files 是纯静态工具类,所有方法都以 Path 为第一参数,语义明确、链式友好,且统一抛出 IOException 或其子类(如 NoSuchFileException)。
String content = Files.readString(path, StandardCharsets.UTF_8);Files.writeString(path, "hello", CREATE, WRITE);
byte[] data = Files.readAllBytes(path);Files.write(path, data, CREATE, TRUNCATE_EXISTING);
Files.copy(src, dst, REPLACE_EXISTING);Files.move(src, dst, ATOMIC_MOVE); // 若支持,为原子操作
Files.createDirectories(path); // 创建多级目录Files.list(path).forEach(System.out::println); // 流式遍历(不递归)Files.walk(path).filter(Files::isRegularFile).forEach(...); // 递归遍历
真正用好 NIO.2,需注意几个关键点:
toString() 当路径字符串 —— 它返回的是内部格式化结果,可能含转义或平台相关表示;应使用 p.toAbsolutePath().normalize().toString() 获取标准路径串Files.exists(p),而非 p.toFile().exists();后者绕过 NIO 语义,且对符号链接行为不一致Files.lines(path) 返回 Stream,内存友好;Files.walk() 支持深度限制和并发处理Files.setPosixFilePermissions(p, perms) 在 Windows 上会抛 UnsupportedOperationException;可用 Files.getAttribute(p, "dos:readonly") 适配 Windows 属性不是功能更多,而是设计更严谨:
Path 分离「路径表示」与「文件操作」,职责单一;File 把两者混在一起,导致方法语义模糊(如 File.renameTo() 返回 boolean 却不说明失败原因)Files 方法全部声明抛出检查异常,强制处理错误场景;而 File 大量方法静默失败(如 mkdirs() 返回 false 却不告诉你为什么)File 基本无法可靠操作这些LinkOption.NOFOLLOW_LINKS 控制,逻辑可控