2010-01-18 105 views
6

我正在调用通过管道连接的命令行程序。所有这一切都适用于Linux。Java exec()不返回管道连接命令的预期结果

我的方法:

protected String execCommand(String command) throws IOException { 
    String line = null; 
    if (command.length() > 0) { 
     Process child = Runtime.getRuntime().exec(command); 
     InputStream lsOut = child.getInputStream(); 
     InputStreamReader r = new InputStreamReader(lsOut); 
     BufferedReader in = new BufferedReader(r); 

     String readline = null; 
     while ((readline = in.readLine()) != null) { 
      line = line + readline; 
     } 
    } 

    return line; 
} 

如果我打电话给一些猫文件| grep asd,我得到了预期的结果。但并非所有的命令都能正常工作比如这个:

cat /proc/cpuinfo | wc -l 

或本:

cat /proc/cpuinfo | grep "model name" | head -n 1 | awk -F":" '{print substr($2, 2, length($2))} 

方法将返回null。我猜这个问题取决于输出格式化命令,如,,wc等我怎么能解决这个问题,并得到最终结果的输出?

+1

无用的(使用* 3)的'cat'。 – 2010-01-18 20:35:01

+0

@丹尼斯 - 的确如此。发现得好! – 2010-01-18 20:39:35

+0

@Bobby我试过了,这没有帮助。 – Pawka 2010-01-18 20:54:11

回答

2

仍然没有找到适当的解决方案来使用Runtime.exec执行管道命令,但找到了解决方法。我只是写了这些脚本来分离bash文件。然后Runtime.exec调用这些bash脚本并获得预期结果。

+0

这是我通过google找到的唯一解决方案。你不能接受你自己的答案?无论如何,非常感谢。 – 2010-11-08 19:05:58

+0

看起来我可以:-) – Pawka 2010-11-12 05:49:50

8

管道(如重定向,或>)是shell的功能,因此直接从Java中删除将不起作用。你需要做的是这样的:

/bin/sh -c "your | piped | commands | here" 

与命令行(包括管道)执行shell进程的-c(引号)后确定。

还要注意的是,你必须消耗stdoutstderr同时,否则你的衍生进程将阻止等待你的过程中要消耗的输出(或错误)。更多信息here

+0

Thx回应,但这仍然不能解决问题。 – Pawka 2010-01-18 20:58:14

2

使用Runtime.exec的所有人都应阅读this

0

快速和肮脏的事情是:

command = "/bin/sh -c '" + command.replaceAll("'", "'\''") + "'" 

通常情况下,你必须注意外壳注射(即有人偷偷"; rm -rf /;"到命令)。但是,如果部分命令可以由其他用户输入提供,那只是一个问题。

缓慢而痛苦的做法是自己用Java做Bash。如果你顺着这条路走,你会发现所有的wonderful things that Bash gives you都不能直接从Process.exec(管道,重定向,复合命令,可变扩展,算术评估...)中获得。

  1. 解析|字符的命令。一定要注意||和引用的字符串。
  2. 为每个管道命令产生一个新的Process
  3. 创建Thread s,读取一个命令的输出并将其写入下一个命令的输入。
0

可能有点为时已晚,但为他人寻找一个解决方案,试试这个...

String[] cmd = { 
         "/bin/sh", 
         "-c", 
         "cat /proc/cpuinfo | wc -l" 
        }; 

Process process = Runtime.getRuntime().exec(cmd); 

所有最好的..