java - 如何在 Java 代码中调用 bash cmd "find -L"

标签 java linux spring bash

我想检查一个目录是否包含任何损坏的链接并返回数字。我想使用命令 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/

相关文章:

java - JDBC 异常 : Operation not allowed after ResultSet closed

java - 基于 Java 的移动操作系统的设备驱动程序

java - sbt 如何知道查看 app 文件夹中的 play 框架源代码?

linux - Bash 将信息从脚本传递到脚本

spring - 在 Controller 内部构造时, Autowiring 服务以命令 Grails 中的对象的正确方法

java - 我想最小化@Transactional 的范围吗?

linux - 将 Linux 移植到 ARM

linux - 永久配置 Bitnami Standalone Linux 版本的静态 IP 地址

spring - 如何为spring sockjs websocket服务器实现增加输出缓冲区

java - 我的 spring 安全代码没有使用静态资源,如 css、js 和图像文件夹