如何从Java程序中在终端运行命令?

40

我需要从JAVA程序中在Fedora 16终端运行一个命令。我尝试使用

Runtime.getRuntime().exec("xterm"); 

但这只是打开了终端,我无法执行任何命令。

我也尝试过以下方法:

OutputStream out = null;
Process proc = new ProcessBuilder("xterm").start();
out = proc.getOutputStream();  
out.write("any command".getBytes());  
out.flush(); 

但是我仅能够打开终端,却无法运行命令。 有什么建议吗?


1
你尝试过 Runtime.getRuntime().exec(<插入命令名称>) 吗?你不需要打开 xterm,因为那是打开你的终端。 - Karthik T
你应该尝试使用 sh -s,然后你可以使用你编写的代码,shell 将会接受来自流的命令,或者使用 sh -c <要运行的命令>,指定在参数中的命令将会被运行。 - ppeterka
6个回答

55
你需要使用 bash 可执行文件来运行它,像这样:
Runtime.getRuntime().exec("/bin/bash -c your_command");

更新: 正如xav建议的那样,更建议使用ProcessBuilder代替:

String[] args = new String[] {"/bin/bash", "-c", "your_command", "with", "args"};
Process proc = new ProcessBuilder(args).start();

6
现在不鼓励使用Runtime.exec(),应该使用ProcessBuilder代替(http://docs.oracle.com/javase/7/docs/api/java/lang/Runtime.html#exec%28java.lang.String[],%20java.lang.String[],%20java.io.File%29)。 - xav

22
我支持Karthik T的回答。您不需要打开终端来运行命令。
例如,
// file: RunShellCommandFromJava.java
import java.io.BufferedReader;
import java.io.InputStreamReader;

public class RunShellCommandFromJava {

    public static void main(String[] args) {

        String command = "ping -c 3 www.google.com";

        Process proc = Runtime.getRuntime().exec(command);

        // Read the output

        BufferedReader reader =  
              new BufferedReader(new InputStreamReader(proc.getInputStream()));

        String line = "";
        while((line = reader.readLine()) != null) {
            System.out.print(line + "\n");
        }

        proc.waitFor();   

    }
} 

输出结果:
$ javac RunShellCommandFromJava.java
$ java RunShellCommandFromJava
PING http://google.com (123.125.81.12): 56 data bytes
64 bytes from 123.125.81.12: icmp_seq=0 ttl=59 time=108.771 ms
64 bytes from 123.125.81.12: icmp_seq=1 ttl=59 time=119.601 ms
64 bytes from 123.125.81.12: icmp_seq=2 ttl=59 time=11.004 ms

--- http://google.com ping statistics ---
3 packets transmitted, 3 packets received, 0.0% packet loss
round-trip min/avg/max/stddev = 11.004/79.792/119.601/48.841 ms

但是,如果命令执行的结果是一个很长的日志,那么这个readLine就完全不起作用了,如何读取它呢? - Yash Agrawal
@YashAgrawal,你能提供生成大量日志的命令(或类似命令),这样我就可以再次测试这个程序吗? - shyan1
我无法在此处发布它,因为运行Hadoop命令时生成的日志非常长,例如:/hadoop.../bin/hdfs jar some.jar /input /output。 - Yash Agrawal
@YashAgrawal 我刚好发现了一个与你遇到的情况基本相同的问题https://dev59.com/R2025IYBdhLWcg3w76lq。我想你也许想要查看一下,看看它是否有帮助? - shyan1

11

您实际上不需要从xterm会话中运行命令,您可以直接运行它:

String[] arguments = new String[] {"/path/to/executable", "arg0", "arg1", "etc"};
Process proc = new ProcessBuilder(arguments).start();

如果进程对输入流进行交互响应,并且您想注入值,则应像以前一样操作:

OutputStream out = proc.getOutputStream();  
out.write("command\n");  
out.flush();

不要忘记在结尾加上'\n',因为大多数应用程序会使用它来确定单个命令输入的结尾。


1
可执行文件可能接受的任何命令行参数(如果有)。 - Chris Cooper

7

正如其他人所说,您可以在不使用xterm的情况下运行外部程序。但是,如果您想在终端窗口中运行它,例如让用户与其交互,xterm允许您指定要作为参数运行的程序。

xterm -e any command

在 Java 代码中,这就变成了:
String[] command = { "xterm", "-e", "my", "command", "with", "parameters" };
Runtime.getRuntime().exec(command);

或者,使用ProcessBuilder:

String[] command = { "xterm", "-e", "my", "command", "with", "parameters" };
Process proc = new ProcessBuilder(command).start();

现在不建议使用Runtime.exec()方法,应该使用ProcessBuilder代替(http://docs.oracle.com/javase/7/docs/api/java/lang/Runtime.html#exec%28java.lang.String[],%20java.lang.String[],%20java.io.File%29)。 - xav
编辑 ProcessBuilder 变体。文档中的注释提到 ProcessBuilder 是首选的启动“修改环境”的进程的方法,但在本例中没有使用。无论如何,我的答案的主要补充是包括如何在 xterm 中运行命令。 - user1252434
感谢您的回答。我需要从Java运行一个makefile。当我运行命令make时,它返回TERM环境变量未设置。但是当我运行命令xterm -e make时,它就像魔法一样正常工作了。 - Ido

4
我不知道为什么,但是 "/bin/bash" 版本对我来说没有起作用。相反,更简单的版本按照 Oracle 文档中给出的示例运行良好。
String[] args = new String[] {"ping", "www.google.com"};
Process proc = new ProcessBuilder(args).start();

3
我知道这个问题已经很老了,但是这里有一个库可以封装ProcessBuilder api。链接

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接