尝试将文件从子目录以及结构移动到父目录。并且无法使用Files.move()
完成此操作。为了说明问题,请参见下面的目录结构。
$ tree
.
└── b
├── c
│ ├── cfile.gtxgt
│ └── d
│ ├── dfile.txt
│ └── e
└── x
└── y
└── z
├── 2.txt
└── p
├── file1.txt
└── q
├── file
├── file2.txt
└── r
└── 123.txt
我想通过Java模拟下面的移动命令。
$ mv b/x/y/z/* b/c
b/x/y/z/2.txt -> b/c/2.txt
b/x/y/z/p -> b/c/p
输出应该类似于
$ tree
.
└── b
├── c
│ ├── 2.txt
│ ├── cfile.gtxgt
│ ├── d
│ │ ├── dfile.txt
│ │ └── e
│ └── p
│ ├── file1.txt
│ └── q
│ ├── file
│ ├── file2.txt
│ └── r
│ └── 123.txt
└── x
└── y
└── z
在此移动中,目录z
下的所有文件和目录都已移动到c
。
我尝试这样做:
public static void main(String[] args) throws IOException{
String aPath = "/tmp/test/a/";
String relativePathTomove = "b/x/y/z/";
String relativePathToMoveTo = "b/c";
Files.move(Paths.get(aPath, relativePathTomove), Paths.get(aPath, relativePathToMoveTo), StandardCopyOption.REPLACE_EXISTING);
}
但是,这将导致抛出的java.nio.file.DirectoryNotEmptyException: /tmp/test/a/b/c
异常,并且如果将REPLACE_EXISTING
选项删除,则代码将抛出java.nio.file.FileAlreadyExistsException: /tmp/test/a/b/c
。
此question的答案使用递归函数来解决此问题。但就我而言,这将涉及更多的复杂性,因为我什至需要在新位置重新创建子目录结构。
我没有尝试使用commons-io
实用程序方法org.apache.commons.io.FileUtils#moveDirectoryToDirectory
的选项,因为此代码似乎是先复制文件,然后再从原始位置删除它们。就我而言,文件很大,因此这不是首选。
如何在不诉诸复制的情况下实现Java中的move功能。单个文件是我唯一的选择吗?
TLDR:如何模拟Java中的mv
功能,以将带有文件和结构的子目录移动到父目录。
答案 0 :(得分:-1)
我最终这样做:
创建一个FileVisitor
实现,如下所示:
package com.test.files;
import org.apache.log4j.Logger;
import javax.validation.constraints.NotNull;
import java.io.IOException;
import java.nio.file.FileVisitResult;
import java.nio.file.FileVisitor;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.attribute.BasicFileAttributes;
import java.util.Objects;
import static java.nio.file.FileVisitResult.TERMINATE;
public class MoveFileVisitor implements FileVisitor<Path> {
private static final Logger LOGGER = Logger.getLogger(MoveFileVisitor.class);
private final Path target;
private final Path source;
public MoveFileVisitor(@NotNull Path source, @NotNull Path target) {
this.target = Objects.requireNonNull(target);
this.source = Objects.requireNonNull(source);
}
@Override
public FileVisitResult preVisitDirectory(Path dir, BasicFileAttributes attrs) throws IOException {
Path relativePath = source.relativize(dir);
Path finalPath = target.resolve(relativePath);
Files.createDirectories(finalPath);
return FileVisitResult.CONTINUE;
}
@Override
public FileVisitResult visitFile(Path file, BasicFileAttributes attrs) throws IOException {
Path relativePath = source.relativize(file);
Path finalLocation = target.resolve(relativePath);
Files.move(file, finalLocation);
return FileVisitResult.CONTINUE;
}
@Override
public FileVisitResult visitFileFailed(Path file, IOException exc) {
LOGGER.error("Failed to visit file during move" + file.toAbsolutePath(), exc);
return TERMINATE;
}
@Override
public FileVisitResult postVisitDirectory(Path dir, IOException exc) throws IOException {
Files.delete(dir);
return FileVisitResult.CONTINUE;
}
}
然后像这样与这个访问者一起走路:
String source = "/temp/test/a/b/x/y/z";
String target = "/temp/test/a/b/c";
MoveFileVisitor visitor = new MoveFileVisitor(Paths.get(source), Paths.get(target));
Files.walkFileTree(Paths.get(source), visitor);