Java - проблема с несколькими одновременными runtime.exec () InputStreams - PullRequest
7 голосов
/ 21 января 2011

У меня нет выбора, кроме как извлечь некоторые внешние данные с помощью нескольких Runtime.exec() вызовов VBScript. Я действительно ненавижу эту реализацию, потому что теряю свою межплатформенную гибкость, но в конечном итоге я могу разработать похожие * nix-скрипты, чтобы хотя бы смягчить проблему. Прежде чем кто-либо спросит, я не могу обойти необходимость вызова внешнего скрипта для сбора моих данных. Я буду жить с проблемами, которые вызывают.

Процессы exec() выполняются в пользовательском классе, который расширяет Runnable. Для чтения данных из getInputStream().

используется BufferedReader.

Редактировать : добавлено больше кода в соответствии с запросом, но я не понимаю, насколько уместен дополнительный код :) Надеюсь, это поможет, потому что форматирование заняло некоторое время! Да, и спокойно относитесь к моему стилю кода, если он уродлив, но конструктивная критика приветствуется ...

public class X extends JFrame implements Runnable {

   ...
   static final int THREADS_MAX = 4;
   ExecutorService  exec;
   ...
   public static void main(String[] args) {
      ...
      SwingUtilities.invokeLater(new X("X"));
   } // End main(String[])

   public X (String title) {
      ...
      exec = Executors.newFixedThreadPool(THREADS_MAX);
      ...

      // Create all needed instances of Y
      for (int i = 0; i < objects.length; i++) {
         Y[i] = new Y(i);
      } // End for(i)

      // Initialization moved here for easy single-thread testing
      // Undesired, of course
      for (int i = 0; i < objects.length; i++) {
         Y[i].initialize(parent);
      } // End for(i)

   } // End X

   class Y implements Runnable {
      // Define variables/arrays used to capture data here
      String computerName = "";
      ...

      public Y(int rowIndex) {
         row          = rowIndex;
         ...
         computerName = (String)JTable.getValueAt(row, 0);
         ...
         exec.execute(this);
      } // End Y(int)

      public void run() {
         // Initialize variables/arrays used to capture data here
         ...

         // Initialization should be done here for proper threading
         //initialize(parent);
      } // End run()

      public void initialize(Z obj) {
         runTime = Runtime.getRuntime();
         ...

         try {
            process = runTime.exec("cscript.exe query.vbs " + computerName);
            stdErr  = process.getErrorStream();
            stdIn   = process.getInputStream();
            isrErr  = new InputStreamReader(stdErr);
            isrIn   = new InputStreamReader(stdIn);
            brErr   = new BufferedReader(isrErr);
            brIn    = new BufferedReader(isrIn);

            while ((line = brIn.readLine()) != null) {
               // Capture, parse, and store data here
               ...
            } // End while

         } catch (IOException e) {
            System.out.println("Unable to run script");
         } catch (Exception e) {
            e.printStackTrace();
         } finally {
            try {
               stdErr.close();
               stdIn. close();
               isrErr.close();
               isrIn. close();
               brErr. close();
               brIn.  close();
            } catch (IOException e) {
               System.out.println("Unable to close streams.");
            } // End try
         } // End try
      } // End initialize(Z)
      ...
   } // End class Y
} // End class X

Если я выполняю команды индивидуально, я собираю данные, как я ожидаю. Однако, если я выполняю команды в блоке run() класса (то есть вызовы являются параллельными, как я и надеюсь), создается впечатление, что генерируется только один входной поток, который все BufferedReaders используют одновременно.

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

exec 0: Line1
exec 1: Line1
exec 2: Line1
exec 0: Line2
exec 1: Line2
exec 2: Line2
exec 0: Line3
exec 1: Line3
exec 2: Line3
...

Что странно, так это то, что я получаю ожидаемое количество экземпляров самой первой строки выходных данных (Microsoft (R) Windows Script Host Version 5.7), но после этой строки только один процесс продолжает генерировать данные во входном потоке, и все читатели произвольно потребляют этот один поток, такой как этот пример:

exec 2: Microsoft (R) Windows Script Host Version 5.7
exec 0: Microsoft (R) Windows Script Host Version 5.7
exec 1: Microsoft (R) Windows Script Host Version 5.7
exec 0: line2
exec 1: line3
exec 2: line4
...

Что еще хуже, читатели останавливаются, и readLine() никогда не возвращает ноль. Я читал, что этот тип поведения может иметь отношение к размеру буфера, но когда я запускаю только два параллельных потока, даже с коротким выводом, он все равно демонстрирует то же поведение. В stdErr ничего не записано, чтобы указать, что есть проблема.

Чтобы проверить, не является ли это ограничением хоста скрипта, я создал пакетный файл, который одновременно START копирует несколько экземпляров скрипта. Я должен заявить, что он был запущен за пределами Java в оболочке cmd и запускает несколько собственных оболочек. Однако каждый параллельный экземпляр полностью возвращал ожидаемые результаты и вел себя хорошо.

Редактировать: В качестве еще одной идеи устранения неполадок, я решил снова включить параллелизм, но пошатнуть мой метод инициализации, вставив следующее в мой Y.run() блок:

try {
   Thread.sleep((int)(Math.random() * 1200));
} catch (InterruptedException e) {
   System.out.println("Can't sleep!");
} // End try
initialize(monitor);

в мой код. Я начинаю видеть несколько выходных данных для первых нескольких строк, но он быстро возвращается к нескольким потребителям, потребляющим одного и того же производителя, и как только первый завершенный поток закрывается, остальные потребители запускают исключения. Следующий потребитель стреляет IOException: Read error, а остальные стреляют IOException: Stream closed!

Согласно Мааартину, возможно запускать несколько одновременных InputStreams, поэтому теперь возникает вопрос, что вызывает нежелательное поведение? Как я могу независимо захватить их входные потоки? Я не хочу писать во временный файл только для того, чтобы обработать данные обратно, если я могу избежать этого.

Ответы [ 2 ]

5 голосов
/ 21 января 2011

Я думаю, вы должны быть осторожны с областью действия переменных ввода-вывода.Вот быстрый код, который отлично работает с параллельными входными потоками из 4 дочерних процессов ...

import java.io.*;

public class MultiExec {

        private final static String[] comLines = {
                        "date",
                        "ls /var/spool/postfix",
                        "ls -F /usr/local/bin",
                        "wc -l /etc/apache2/apache2.conf"};

        public void execute() {
                for (int i = 0 ; i < comLines.length ; i++) {
                        ExecutableChild ec = new ExecutableChild (i, comLines[i]);
                        new Thread (ec).start();
        }}

        public class ExecutableChild implements Runnable {

                private int prIndex;
                private String executable;

                public ExecutableChild (int k, String cmd) {
                        prIndex = k;
                        executable = cmd;
                }

                public void run () {
                        try {
                                Process child = Runtime.getRuntime().exec(executable);
                                BufferedReader br = new BufferedReader (new InputStreamReader (
                                                                child.getInputStream()));
                                for (String s = br.readLine() ; s != null ; s = br.readLine())
                                        System.out.println ("[" + prIndex + "] " + s);
                                br.close();
                        } catch (IOException ioex) {
                                System.err.println ("IOException for process #"+
                                                prIndex+ ": " + ioex.getMessage());
        }}}

        public static void main (String[] args) {
                new MultiExec().execute();
        }
}

Вывод из приведенного выше кода (% javac MultiExec.java; java MultiExec)

[2] tomcat*
[0] Thu Jan 20 18:38:31 CST 2011
[3] 368 /etc/apache2/apache2.conf
[1] active
[1] bounce
[1] corrupt
[1] defer
[1] deferred
[1] etc
[1] flush
[1] hold
[1] incoming
[1] lib
[1] maildrop
[1] pid
[1] private
[1] public
[1] saved
[1] trace
[1] usr
[1] var

Если вы получите нам исходный код вашей попытки, мы можем обсудить это.С наилучшими пожеланиями, - MS

========================================================================================

Редактировать: DN: Я понимаюВаши опасения по поводу 1-строчных выходов.Давайте сделаем небольшой скрипт ...

#!/usr/bin/perl -w
foreach (1..50) {
        print "$_\n";
}

и отредактированную версию вышеупомянутого Java-кода ... comLines изменились, и Thread.sleep добавлен после каждого println ()

public class MultiExec {

        private final static String[] comLines = {
                        "ls /var/spool/postfix",
                        "perl count50.pl",
                        "cat MultiExec.java",
                        "head -40 /etc/apache2/apache2.conf"};

        public void execute() {
                for (int i = 0 ; i < comLines.length ; i++) {
                        ExecutableChild ec = new ExecutableChild (i, comLines[i]);
                        new Thread (ec).start();
        }}

        public class ExecutableChild implements Runnable {

                private int prIndex;
                private String executable;

                public ExecutableChild (int k, String cmd) {
                        prIndex = k;
                        executable = cmd;
                }

                public void run () {
                        try {
                                Process child = Runtime.getRuntime().exec(executable);
                                BufferedReader br = new BufferedReader (new InputStreamReader (
                                                                child.getInputStream()));
                                for (String s = br.readLine() ; s != null ; s = br.readLine()) {
                                        System.out.println ("[" + prIndex + "] " + s);
                                        try {
                                                Thread.sleep (20);
                                        } catch (InterruptedException intex) {
                                }}
                                br.close();
                        } catch (IOException ioex) {
                                System.err.println ("IOException for process #"+
                                                                prIndex+ ": " + ioex.getMessage());
        }}}

        public static void main (String[] args) {
                new MultiExec().execute();
}}

Вот вывод сейчас (после компиляции / запуска) ...

[0] active
[1] 1
[2] import java.io.*;
[3] #
[2]
[0] bounce
[1] 2
[3] # Based upon the NCSA server configuration files originally by Rob McCool.
[2] public class MultiExec {
[1] 3
[0] corrupt
[3] #
[1] 4
[2]
[0] defer
[3] # This is the main Apache server configuration file.  It contains the
[2]     private final static String[] comLines = {
[0] deferred
[1] 5
[3] # configuration directives that give the server its instructions.
[2]                     "ls /var/spool/postfix",
[0] etc
[1] 6
[3] # See http://httpd.apache.org/docs/2.2/ for detailed information about
[2]                     "perl count50.pl",
[0] flush
[1] 7
[3] # the directives.
[2]                     "cat MultiExec.java",
[1] 8
[0] hold
[3] #
[1] 9
[2]                     "head -40 /etc/apache2/apache2.conf"};
[0] incoming
[3] # Do NOT simply read the instructions in here without understanding
[2]
[0] lib
[1] 10
[3] # what they do.  They're here only as hints or reminders.  If you are unsure
[1] 11
[2]     public void execute() {
[0] maildrop
[3] # consult the online docs. You have been warned.
[2]             for (int i = 0 ; i < comLines.length ; i++) {
[0] pid
[1] 12
[3] #
[1] 13
[2]                     ExecutableChild ec = new ExecutableChild (i, comLines[i]);
[0] private
[3] # The configuration directives are grouped into three basic sections:
[1] 14
[2]                     new Thread (ec).start();
[0] public
[3] #  1. Directives that control the operation of the Apache server process as a
[2]     }}
[1] 15
[0] saved
[3] #     whole (the 'global environment').
[1] 16
[0] trace
[2]
[3] #  2. Directives that define the parameters of the 'main' or 'default' server,
[0] usr
[2]     public class ExecutableChild implements Runnable {
[1] 17
[3] #     which responds to requests that aren't handled by a virtual host.
[0] var
[2]
[1] 18
[3] #     These directives also provide default values for the settings
[1] 19
[2]             private int prIndex;
[3] #     of all virtual hosts.
[1] 20
[2]             private String executable;
[3] #  3. Settings for virtual hosts, which allow Web requests to be sent to
[2]
[1] 21
[3] #     different IP addresses or hostnames and have them handled by the
[1] 22
[2]             public ExecutableChild (int k, String cmd) {
[3] #     same Apache server process.
[1] 23
[2]                     prIndex = k;
[3] #
[1] 24
[2]                     executable = cmd;
[3] # Configuration and logfile names: If the filenames you specify for many
[2]             }
[1] 25
[3] # of the server's control files begin with "/" (or "drive:/" for Win32), the
[2]
[1] 26
[3] # server will use that explicit path.  If the filenames do *not* begin
[1] 27
[2]             public void run () {
[3] # with "/", the value of ServerRoot is prepended -- so "/var/log/apache2/foo.log"
[1] 28
[2]                     try {
[3] # with ServerRoot set to "" will be interpreted by the
[1] 29
[2]                             Process child = Runtime.getRuntime().exec(executable);
[3] # server as "//var/log/apache2/foo.log".
[1] 30
[2]                             BufferedReader br = new BufferedReader (new InputStreamReader (
[3] #
[1] 31
[2]                                                             child.getInputStream()));
[3]
[1] 32
[2]                             for (String s = br.readLine() ; s != null ; s = br.readLine()) {
[3] ### Section 1: Global Environment
[1] 33
[2]                                     System.out.println ("[" + prIndex + "] " + s);
[3] #
[1] 34
[2]                                     try {
[3] # The directives in this section affect the overall operation of Apache,
[1] 35
[2]                                             Thread.sleep (20);
[3] # such as the number of concurrent requests it can handle or where it

......

Входные потоки работают просто отлично, не думаю, что у меня естьпроблема здесь.Извините за такой длинный ответ.Желаю вам всего наилучшего и ждем, чтобы увидеть ваш код, - MS

3 голосов
/ 21 января 2011

Убедитесь, что вы указали stdErr и stdIn в правильной области.В этом случае вам нужно объявить их в Y.

Если вы объявляете их в X, каждый раз, когда вы запускаете следующий код:

stdErr  = process.getErrorStream();
stdIn   = process.getInputStream();

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

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