移動文件樹是復制和刪除的文件樹的結合。實際上,有兩種方式來完成文件的移動。一種是使用Files.move(), Files.copy(), 和Files.delete() 這三個方法;另一種是只使用Files.copy(),Files.delete()方法?;谀愕倪x擇,在實現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 + "]"); } }
新聞熱點
疑難解答