2010-01-18 51 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); 

所有最好的..