0

我的目标是使用 mysql 命令恢复 Mysql 转储。

我在使用 ProcessBuilder 和 Apache Commons Exec (1.3) 的代码之间有明显的行为差异。

这段代码工作得很好

List<String> params = new ArrayList<>();
params.add("sh");
params.add("-c");
params.add("mysql -uroot -proot < /tmp/the_dump.sql");

ProcessBuilder pb = new ProcessBuilder(params);
Process p = pb.redirectErrorStream(true).start();
int exitValue = p.waitFor();

但是这个(使用 Apache Commons Exec)

CommandLine cmdLine = new CommandLine("/usr/bin/sh");
cmdLine.addArgument("-c");
cmdLine.addArgument("mysql -uroot -proot -e 'source /tmp/the_dump.sql'");

PumpStreamHandler streamHandler = new PumpStreamHandler(System.out, System.err, System.in);
Executor executor = new DefaultExecutor();
executor.setStreamHandler(streamHandler);
int exitValue = executor.execute(cmdLine);

因以下错误而崩溃:

Apache Commons Exec 出错

请注意 Apache Commons Exec 解释的命令末尾缺少的 '。 命令行解释

有人知道如何使其与 Apache Commons Exec 一起工作吗?

非常感谢你的帮助,西尔万

4

1 回答 1

0

sh shell 被解释mysql -uroot -proot -e 'source /tmp/the_dump.sql'为单个文件并试图找到要执行的文件。您需要通过写入其输入流将命令传递给 sh。

sh shell 通常接受来自标准输入流的命令。如果您在终端窗口中启动 sh,则键盘是标准输入。在这种情况下,您正在启动一个子流程,因此 Java 可以访问该子流程的输入和输出流。在 Java 中,您可以通过执行以下操作将命令写入 sh 的流:

CommandLine cmdLine = new CommandLine("/usr/bin/sh");
byte[] buf = "mysql -uroot -proot -e 'source /tmp/the_dump.sql'".getBytes();
ByteArrayInputStream is = new ByteArrayInputStream(buf);
PumpStreamHandler streamHandler = new PumpStreamHandler(System.out, System.err, is);
Executor executor = new DefaultExecutor();
executor.setStreamHandler(streamHandler);
int exitValue = executor.execute(cmdLine);

请注意,通过在 PumpStreamHandler 中使用System.outSystem.err,您将处理 java 进程的输出和错误流,而不是 sh 子进程。

于 2021-04-02T21:14:41.153 回答