Runtime.getRuntime().exec(cmd)挂起

use*_*404 13 java process runtime.exec processbuilder

我正在执行一个命令,它返回一个文件的修订号; '文件名'.但是如果执行命令时出现问题,则应用程序会挂起.我该怎么做才能避免这种情况?请在下面找到我的代码.

String cmd= "cmd /C si viewhistory --fields=revision --project="+fileName; 
Process p = Runtime.getRuntime().exec(cmd) ;  
BufferedReader in = new BufferedReader(new InputStreamReader(p.getInputStream()));  
String line = null; 
while ((line = in.readLine()) != null) {  
System.out.println(line);  
} 

} catch (Exception e) {  
e.printStackTrace();  
 }
Run Code Online (Sandbox Code Playgroud)

Arh*_*ham 31

我想问题是你只是在阅读InputStream而不是读ErrorStream.您还必须注意并行读取两个流.可能会发生这样的情况:当前从输出流传输的数据填满了操作系统缓冲区,您的exec命令将自动暂停,以便您的读者有机会清空缓冲区.但该程序仍将等待输出处理.因此,发生了挂起.

您可以创建一个单独的类来处理输入和错误流,如下所示,

public class ReadStream implements Runnable {
    String name;
    InputStream is;
    Thread thread;      
    public ReadStream(String name, InputStream is) {
        this.name = name;
        this.is = is;
    }       
    public void start () {
        thread = new Thread (this);
        thread.start ();
    }       
    public void run () {
        try {
            InputStreamReader isr = new InputStreamReader (is);
            BufferedReader br = new BufferedReader (isr);   
            while (true) {
                String s = br.readLine ();
                if (s == null) break;
                System.out.println ("[" + name + "] " + s);
            }
            is.close ();    
        } catch (Exception ex) {
            System.out.println ("Problem reading stream " + name + "... :" + ex);
            ex.printStackTrace ();
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

你使用它的方式如下,

String cmd= "cmd /C si viewhistory --fields=revision --project="+fileName; 
Process p = Runtime.getRuntime().exec(cmd) ;  
s1 = new ReadStream("stdin", p.getInputStream ());
s2 = new ReadStream("stderr", p.getErrorStream ());
s1.start ();
s2.start ();
p.waitFor();        
} catch (Exception e) {  
e.printStackTrace();  
} finally {
    if(p != null)
        p.destroy();
}
Run Code Online (Sandbox Code Playgroud)


mik*_*lly 8

此代码基于相同的想法 Arham's answer,但使用 java 8 并行流实现,这使其更加简洁。

public static String getOutputFromProgram(String program) throws IOException {
    Process proc = Runtime.getRuntime().exec(program);
    return Stream.of(proc.getErrorStream(), proc.getInputStream()).parallel().map((InputStream isForOutput) -> {
        StringBuilder output = new StringBuilder();
        try (BufferedReader br = new BufferedReader(new InputStreamReader(isForOutput))) {
            String line;
            while ((line = br.readLine()) != null) {
                output.append(line);
                output.append("\n");
            }
        } catch (IOException e) {
            throw new RuntimeException(e);
        }
        return output;
    }).collect(Collectors.joining());
}
Run Code Online (Sandbox Code Playgroud)

你可以像这样调用方法

getOutputFromProgram("cmd /C si viewhistory --fields=revision --project="+fileName);
Run Code Online (Sandbox Code Playgroud)

请注意,如果您正在调用的程序挂起,则此方法将挂起,如果需要输入,则会发生这种情况。