Захват stdout при вызове Runtime.exec - PullRequest
54 голосов
/ 19 мая 2009

При возникновении проблем с сетью на клиентских компьютерах я хотел бы иметь возможность запустить несколько командных строк и отправить их результаты по электронной почте.

Я обнаружил, что Runtime.exec позволяет мне выполнять произвольные команды, но более интересным является сбор результатов в строке.

Я понимаю, что могу перенаправить вывод в файл, а затем прочитать его, но мое чувство пауков подсказывает мне, что есть более элегантный способ сделать это.

Предложения

Ответы [ 8 ]

47 голосов
/ 19 мая 2009

Вам нужно захватить как std out, так и std err в процессе. Затем вы можете записать стандартный вывод в файл / почту или аналогичный файл.

См. в этой статье для получения дополнительной информации и, в частности, обратите внимание на механизм StreamGobbler, который захватывает stdout / err в отдельных потоках. Это важно для предотвращения блокировки и является источником многочисленных ошибок, если вы не делаете это правильно!

13 голосов
/ 19 мая 2009

Использование ProcessBuilder . После вызова start () вы получите объект Process , из которого вы можете получить потоки stderr и stdout.

ОБНОВЛЕНИЕ: ProcessBuilder дает вам больше контроля; Вы не должны использовать это, но я нахожу это легче в долгосрочной перспективе. Особенно возможность перенаправить stderr в stdout, что означает, что вам нужно только сосать один поток.

7 голосов
/ 19 мая 2009

Используйте Plexus Utils , оно используется Maven для выполнения всех внешних процессов.

Commandline commandLine = new Commandline();
commandLine.setExecutable(executable.getAbsolutePath());

Collection<String> args = getArguments();

for (String arg : args) {
    Arg _arg = commandLine.createArg();
    _arg.setValue(arg);
}

WriterStreamConsumer systemOut = new WriterStreamConsumer(console);
WriterStreamConsumer systemErr = new WriterStreamConsumer(console);

returnCode = CommandLineUtils.executeCommandLine(commandLine, systemOut, systemErr, 10);
if (returnCode != 0) {
    // bad
} else {
    // good
}
5 голосов
/ 25 июня 2014

Для процессов, которые не генерируют много выходных данных, я думаю, что этого простого решения, использующего Apache IOUtils , достаточно:

Process p = Runtime.getRuntime().exec("script");
p.waitFor();
String output = IOUtils.toString(p.getInputStream());
String errorOutput = IOUtils.toString(p.getErrorStream());

Предупреждение: однако, если ваш процесс генерирует много выходных данных, этот подход может вызвать проблемы, как указано в Класс процесса JavaDoc :

Созданный подпроцесс не имеет собственного терминала или консоли. Все его стандартные операции io (т.е. stdin, stdout, stderr) будут перенаправлены на родительский процесс через три потока (getOutputStream (), getInputStream (), getErrorStream ()). Родительский процесс использует эти потоки для подачи входных данных и получения выходных данных из подпроцесса. Поскольку некоторые собственные платформы предоставляют ограниченный размер буфера только для стандартных входных и выходных потоков, невозможность оперативной записи входного потока или чтения выходного потока подпроцесса может привести к блокировке подпроцесса и даже к тупиковой ситуации.

3 голосов
/ 23 сентября 2013

Это мой вспомогательный класс, который используется годами. Один маленький класс. У этого есть класс streamwobbler JavaWorld, чтобы исправить утечки ресурса JVM. Не знаю, если все еще действует для JVM6 и JVM7, но не повредит. Помощник может прочитать буфер вывода для последующего использования.

import java.io.*;

/**
 * Execute external process and optionally read output buffer.
 */
public class ShellExec {
    private int exitCode;
    private boolean readOutput, readError;
    private StreamGobbler errorGobbler, outputGobbler;

    public ShellExec() { 
        this(false, false);
    }

    public ShellExec(boolean readOutput, boolean readError) {
        this.readOutput = readOutput;
        this.readError = readError;
    }

    /**
     * Execute a command.
     * @param command   command ("c:/some/folder/script.bat" or "some/folder/script.sh")
     * @param workdir   working directory or NULL to use command folder
     * @param wait  wait for process to end
     * @param args  0..n command line arguments
     * @return  process exit code
     */
    public int execute(String command, String workdir, boolean wait, String...args) throws IOException {
        String[] cmdArr;
        if (args != null && args.length > 0) {
            cmdArr = new String[1+args.length];
            cmdArr[0] = command;
            System.arraycopy(args, 0, cmdArr, 1, args.length);
        } else {
            cmdArr = new String[] { command };
        }

        ProcessBuilder pb =  new ProcessBuilder(cmdArr);
        File workingDir = (workdir==null ? new File(command).getParentFile() : new File(workdir) );
        pb.directory(workingDir);

        Process process = pb.start();

        // Consume streams, older jvm's had a memory leak if streams were not read,
        // some other jvm+OS combinations may block unless streams are consumed.
        errorGobbler  = new StreamGobbler(process.getErrorStream(), readError);
        outputGobbler = new StreamGobbler(process.getInputStream(), readOutput);
        errorGobbler.start();
        outputGobbler.start();

        exitCode = 0;
        if (wait) {
            try { 
                process.waitFor();
                exitCode = process.exitValue();                 
            } catch (InterruptedException ex) { }
        }
        return exitCode;
    }   

    public int getExitCode() {
        return exitCode;
    }

    public boolean isOutputCompleted() {
        return (outputGobbler != null ? outputGobbler.isCompleted() : false);
    }

    public boolean isErrorCompleted() {
        return (errorGobbler != null ? errorGobbler.isCompleted() : false);
    }

    public String getOutput() {
        return (outputGobbler != null ? outputGobbler.getOutput() : null);        
    }

    public String getError() {
        return (errorGobbler != null ? errorGobbler.getOutput() : null);        
    }

//********************************************
//********************************************    

    /**
     * StreamGobbler reads inputstream to "gobble" it.
     * This is used by Executor class when running 
     * a commandline applications. Gobblers must read/purge
     * INSTR and ERRSTR process streams.
     * http://www.javaworld.com/javaworld/jw-12-2000/jw-1229-traps.html?page=4
     */
    private class StreamGobbler extends Thread {
        private InputStream is;
        private StringBuilder output;
        private volatile boolean completed; // mark volatile to guarantee a thread safety

        public StreamGobbler(InputStream is, boolean readStream) {
            this.is = is;
            this.output = (readStream ? new StringBuilder(256) : null);
        }

        public void run() {
            completed = false;
            try {
                String NL = System.getProperty("line.separator", "\r\n");

                InputStreamReader isr = new InputStreamReader(is);
                BufferedReader br = new BufferedReader(isr);
                String line;
                while ( (line = br.readLine()) != null) {
                    if (output != null)
                        output.append(line + NL); 
                }
            } catch (IOException ex) {
                // ex.printStackTrace();
            }
            completed = true;
        }

        /**
         * Get inputstream buffer or null if stream
         * was not consumed.
         * @return
         */
        public String getOutput() {
            return (output != null ? output.toString() : null);
        }

        /**
         * Is input stream completed.
         * @return
         */
        public boolean isCompleted() {
            return completed;
        }

    }

}

Вот пример чтения вывода из скрипта .vbs, но аналогичные работы для скриптов linux sh.

   ShellExec exec = new ShellExec(true, false);
   exec.execute("cscript.exe", null, true,
      "//Nologo",
      "//B",            // batch mode, no prompts
      "//T:320",        // timeout seconds
      "c:/my/script/test1.vbs",  // unix path delim works for script.exe
      "script arg 1",
      "script arg 2",
   );
   System.out.println(exec.getOutput());
2 голосов
/ 02 января 2013

VerboseProcess служебный класс от jcabi-log может помочь вам:

String output = new VerboseProcess(
  new ProcessBuilder("executable with output")
).stdout();

Единственная нужная вам зависимость:

<dependency>
  <groupId>com.jcabi</groupId>
  <artifactId>jcabi-log</artifactId>
  <version>0.7.5</version>
</dependency>
2 голосов
/ 19 мая 2009

Runtime.exec () возвращает объект Process, из которого вы можете извлечь вывод любой команды, которую вы выполнили.

1 голос
/ 19 мая 2009

Использование Runtime.exec дает вам процесс. Вы можете использовать getInputStream , чтобы получить стандартный вывод этого процесса, и поместить этот поток ввода в строку, например, через StringBuffer.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...