我想检查一个目录是否包含任何损坏的链接并返回数字。我想使用命令 find -L/dir/* -type d -prune -o -type l | wc -l
它在我的终端上运行良好。但是当我通过 runtime.exec
在我的 java 代码中运行它时,它会抛出一个错误:
无法调用进程“find -L/dir/* -type d -prune -o -type l | wc -l”。原因:查找:路径必须在表达式之前:|用法:查找 [-H] [-L] [-P] [-Olevel] [-D help|tree|search|stat|rates|opt|exec] [path.. .] [表达式]
我尝试了'/dar/*'
、"/dir/*"
和/dir/\*
,但都没有作品。任何帮助表示赞赏。
最佳答案
管道运算符 (|
) 是由 shell 提供的功能,如 bash,而不是由 find
命令提供。 Runtime.exec 只是将所有参数传递给由字符串的第一个单词(在本例中为 find
)定义的命令,而 find
命令不知道管道符号是什么是。
您可以通过在常规终端窗口中运行命令来亲眼看到这种效果,但是 |
转义为 \|
因此它被传递给 find
命令,就像您的 Java 代码所做的那样:
find -L /dir/* -type d -prune -o -type l \| wc -l
一个常见的解决方案是将整个命令作为参数传递给 bash -c
:
Process process = new ProcessBuilder("bash", "-c",
"find -L /dir/* -type d -prune -o -type l | wc -l").start();
(Runtime.exec 已过时;它的替代品是 ProcessBuilder,它具有更明确定义的行为。)
更好的解决方案是计算 Java 中的行数,并完全避免管道符号:
Process process = new ProcessBuilder(
"find", "-L", "/dir/*", "-type", "d", "-prune", "-o", "-type", "l").start();
long brokenLinkCount;
try (BufferedReader lineReader =
new BufferedReader(new InputStreamReader(process.getInputStream()))) {
brokenLinkCount = lineReader.lines().count();
}
将每个参数单独传递给 ProcessBuilder 的另一个好处是该命令适用于所有路径,即使是文件名中带有空格或引号的路径。
最后,即使您说过不想走这条路,最好的方法是取消外部命令并在 Java 中执行:
Path dir = Paths.get("/dir");
Stream<Path> brokenLinks =
Files.find(dir, 1, (path, attr) -> attr.isSymbolicLink())
.filter(path -> {
try {
return !Files.exists(Files.readSymbolicLink(path));
} catch (IOException e) {
System.err.println("Couldn't read link " + path + ": " + e);
return false;
}
});
long brokenLinkCount = brokenLinks.count();
关于java - 如何在 Java 代码中调用 bash cmd "find -L",我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40409657/