Java NIO.2 —— 文件或目录移动操作
移动文件树是复制和删除的文件树的结合。实际上,有两种方式来完成文件的移动。一种是使用Files.move()
, Files.copy()
, 和Files.delete() 这三个方法;另一种是只使用
Files.copy(),
Files.delete()方法。基于你的选择,在实现
FileVisitor
接口中对应的方法完成文件树的移动。下面是一些注意事项。
- 在移动任何目录或文件之前,首先要移动目录本身。因为不为空的目录不能被移动。你需要在
preVisitDirectory()
方法中你使用Files.copy()
方法用拷贝文件来代替。 visitFile()
是用来移动每个文件最合适的方式。你一可以使用Files.move()或联合使用
Files.copy()
和Files.delete()方法。
- 当你移动完所有的源目录到目标目录后,使用
Files.delete()
来删除源目录,此时应该已经为空。这些操作应该在postVisitDirectory()方法中完成。
- 当你拷贝目录或文件时,你需要决定使用
REPLACE_EXISTING
和COPY_ATTRIBUTES选项。同时,当你移动文件或目录时,决定是否要使用
ATOMIC_MOVE选项。
- 如果你想保留源目录的属性,你需要在文件移动以后在
postVisitDirectory()做这些操作。一些属性例如
lastModifiedTime
应该在preVisitDirectory()
方法中提取直到在postVisitDirectory()方法中设置好后保存。这是因为你在从源目录移动完文件以后,文件内容已经更改,最初的最近修改时间已经被新的日期覆盖了。
- 如果文件不能访问,你自己来决定来使用
FileVisitResult.CONTINUE
或TERMINATE
选项。 - 你也可以使用
FOLLOW_LINKS
选项来跟踪符号链接文件,需要注意的是,只会移动符号链接本身文件,不会影响到它指向的实际文件。
下面的代码把 C:\rafaelnadal
目录移动到C:\ATP\players\rafaelnafal
目录。
import java.io.IOException; import java.nio.file.FileVisitOption; import java.nio.file.FileVisitResult; import java.nio.file.FileVisitor; import java.nio.file.Files; import java.nio.file.Path; import java.nio.file.Paths; import java.nio.file.attribute.BasicFileAttributes; import java.nio.file.attribute.FileTime; import java.util.EnumSet; import static java.nio.file.StandardCopyOption.REPLACE_EXISTING; import static java.nio.file.StandardCopyOption.COPY_ATTRIBUTES; import static java.nio.file.StandardCopyOption.ATOMIC_MOVE; class MoveTree implements FileVisitor { private final Path moveFrom; private final Path moveTo; static FileTime time = null; public MoveTree(Path moveFrom, Path moveTo) { this.moveFrom = moveFrom; this.moveTo = moveTo; } static void moveSubTree(Path moveFrom, Path moveTo) throws IOException { try { Files.move(moveFrom, moveTo, REPLACE_EXISTING, ATOMIC_MOVE); } catch (IOException e) { System.err.println("Unable to move " + moveFrom + " [" + e + "]"); } } @Override public FileVisitResult postVisitDirectory(Object dir, IOException exc) throws IOException { Path newdir = moveTo.resolve(moveFrom.relativize((Path) dir)); try { Files.setLastModifiedTime(newdir, time); Files.delete((Path) dir); } catch (IOException e) { System.err.println("Unable to copy all attributes to: " + newdir+" [" + e + "]"); } return FileVisitResult.CONTINUE; } @Override public FileVisitResult preVisitDirectory(Object dir, BasicFileAttributes attrs) throws IOException { System.out.println("Move directory: " + (Path) dir); Path newdir = moveTo.resolve(moveFrom.relativize((Path) dir)); try { Files.copy((Path) dir, newdir, REPLACE_EXISTING, COPY_ATTRIBUTES); time = Files.getLastModifiedTime((Path) dir); } catch (IOException e) { System.err.println("Unable to move " + newdir + " [" + e + "]"); return FileVisitResult.SKIP_SUBTREE; } return FileVisitResult.CONTINUE; } @Override public FileVisitResult visitFile(Object file, BasicFileAttributes attrs) throws IOException { System.out.println("Move file: " + (Path) file); moveSubTree((Path) file, moveTo.resolve(moveFrom.relativize((Path) file))); return FileVisitResult.CONTINUE; } @Override public FileVisitResult visitFileFailed(Object file, IOException exc) throws IOException { return FileVisitResult.CONTINUE; } } class Main { public static void main(String[] args) throws IOException { Path moveFrom = Paths.get("C:/rafaelnadal"); Path moveTo = Paths.get("C:/ATP/players/rafaelnadal"); MoveTree walk = new MoveTree(moveFrom, moveTo); EnumSet opts = EnumSet.of(FileVisitOption.FOLLOW_LINKS); Files.walkFileTree(moveFrom, opts, Integer.MAX_VALUE, walk); } }
你也可以使用其他的方法来完成Files.move() 相同的工作,因为每一次移动就是一对拷贝删除操作。所以,你可以重写
moveSubTree()
方法。
static void moveSubTree(Path moveFrom, Path moveTo) throws IOException { try { Files.copy(moveFrom, moveTo, REPLACE_EXISTING, COPY_ATTRIBUTES); Files.delete(moveFrom); } catch (IOException e) { System.err.println("Unable to move " + moveFrom + " [" + e + "]"); } }
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· 分享 3 个 .NET 开源的文件压缩处理库,助力快速实现文件压缩解压功能!
· Ollama——大语言模型本地部署的极速利器
· DeepSeek如何颠覆传统软件测试?测试工程师会被淘汰吗?