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

Posted

技术标签:

【中文标题】如何从java程序在终端运行命令?【英文标题】:how to run a command at terminal from java program? 【发布时间】:2013-02-27 16:21:13 【问题描述】:

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

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

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

我也试过这个:

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

但我仍然只能打开终端,但无法运行命令。 关于如何做到这一点的任何想法?

【问题讨论】:

你试过Runtime.getRuntime().exec(<insert command name here>); 你不需要打开xterm 那就是打开你的终端。 你应该试试sh -s,你可以使用你写的代码,shell会接受来自流的命令,或者sh -c <the command you want to run>,参数中指定的命令将被运行。 【参考方案1】:

我知道这个问题已经很老了,但是这里有一个封装了 ProcessBuilder api 的library。

【讨论】:

【参考方案2】:

我不知道为什么,但由于某种原因,“/bin/bash”版本不适合我。 相反,按照here at Oracle Docs. 给出的示例,更简单的版本起作用了

String[] args = new String[] "ping", "www.google.com";
Process proc = new ProcessBuilder(args).start();

【讨论】:

【参考方案3】:

您需要像这样使用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();

【讨论】:

现在不鼓励使用Runtime.exec():使用应使用 ProcessBuilder (docs.oracle.com/javase/7/docs/api/java/lang/…)【参考方案4】:

正如其他人所说,您可以在没有 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 (docs.oracle.com/javase/7/docs/api/java/lang/…) Edited ProcessBuilder 变体。文档中的注释是指 ProcessBuilder 优先使用“修改的环境”启动进程,但在这种情况下不使用。无论如何,我的回答的主要补充是包含如何在 xterm 中运行命令。 感谢您的回答。我需要从 Java 运行一个 makefile。当我运行命令make 时,它返回TERM environment variable not set.。但是当我运行命令 xterm -e make 时,它就像一个魅力。【参考方案5】:

我投票支持 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的东西根本不起作用,怎么读呢? @YashAgrawal 能否提供生成冗长日志的命令(或类似命令),以便我再次测试该程序? 我无法在此处发布它,因为它在运行 hadoop 命令时的日志非常长,例如:/hadoop.../bin/hdfs jar some.jar /input /output @YashAgrawal 我只是碰巧发现还有一个问题***.com/questions/5711084/… 处理基本相同的情况。我想你可能也想检查一下,你能看看它是否有帮助?【参考方案6】:

您实际上不需要从 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”,因为大多数应用程序会使用它来标识单个命令输入的结尾。

【讨论】:

可执行文件可能采用的任何命令行参数(如果有的话)。

以上是关于如何从java程序在终端运行命令?的主要内容,如果未能解决你的问题,请参考以下文章

如何在 macOS 上通过命令隐藏终端窗口

如何从终端运行Java public static void main [复制]

如何使用swift从Xcode coco app运行像FFMPEG这样的终端程序?

linux如何在终端启动程序后可以继续输入命令?

如何在 MAMP 安装上从终端运行 mysql 命令?

如何从类似于 pip 或 howdoi 命令的任何终端实例将 python 脚本作为全局命令运行?