移动文件树是复制和删除的文件树的结合。实际上,有两种方式来完成文件的移动。一种是使用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 + "]");
}
}