使用ProcessBuilder / Runtime.exec()启动的外部进程在XP上失败,适用于Win 7

时间:2012-02-08 09:44:06

标签: java windows process windows-xp

我正在开发一个Java程序,它必须获取机器序列号,CPU序列号等。在Windows上,WMI接口是查询此类信息的最佳方式,使用命令行查询的标准方法是

wmic bios get serialnumber

产生输出:

SerialNumber
WWV46RT609A3467173E

将其翻译成Java,我使用了Runtime.exec()和ProcessBuilder,如下所示: (评论过程p是我之前做过的)。这里,component和item对应于上面命令行中的'bios'和'serialnumber'。

    String ret = "";
    ProcessBuilder pb = new ProcessBuilder("wmic", component, "get", item);
    pb.redirectErrorStream(true);
    // Process p = Runtime.getRuntime().exec(
    // "wmic " + component + " get " + item);
    Process p = pb.start();
    InputStreamReader isr = new InputStreamReader(p.getInputStream());
    BufferedReader input = new BufferedReader(isr);
    String str;
    while ((str = input.readLine()) != null) {
        if (str.equalsIgnoreCase(item) || StringUtils.isBlank(str)) {
            continue;
        }
        ret = str.trim();
    }
    input.close(); 
    isr.close();
    System.out.println(ret); 

此代码段在Windows 7上完美运行,但在Windows XP上挂起。使用命令行中的wmic可以在两个操作系统上运行。 我读here说处理被调用进程的stdout和stderr都有问题,因此调用了redirectErrorStream()。

为什么它在Windows 7上完美运行但在XP上失败?有没有办法产生一个单独的线程,又名“StreamGobbler”? (链接的示例非常古老,早于ProcessBuilder类,其redirectErrorStream()调用。

2 个答案:

答案 0 :(得分:6)

我希望你现在已经解决了这个问题。如果没有,这就是你需要做的。首先,我也遇到了同样的问题,并发现它是bufferedReader问题。它陷入僵局,导致windows xp挂起。解决方案是通过附加"<NUL"命令来模拟bufferedreader的行尾(eof)。

 String[] command = {"CMD", "/C", "WMIC COMPUTERSYSTEM GET USERNAME <NUL "} and executing this command. 

答案 1 :(得分:0)

您必须使用线程来捕获输出(标准和错误)。

您还可以查看此Apache library