it-swarm.com.ru

Как заставить Java getRuntime (). Exec () запускать программу командной строки с аргументами?

Я пытался написать программу на Java, которая использует метод Runtime.getRuntime().exec() для использования командной строки для запуска экземпляра программы "tesseract".

В некотором смысле, Tesseract - это бесплатная программа с открытым исходным кодом, которая используется для оптического распознавания символов (OCR). Он принимает в файл изображения и выводит текстовый документ. Это программа командной строки, которая использует эту команду для запуска 

(из командной строки командной строки)

tesseract imageFilePath outFilePath [optional arguments] 

пример:

tesseract "C:\Program Files (x86)\Tesseract-OCR\doc\eurotext.tif" "C:\Users\Dreadnought\Documents\TestingFolder\out"

первый аргумент вызывает программу tesseract, второй - абсолютный путь к файлу изображения, а последний аргумент - путь и имя того, каким должен быть выходной файл. Для Tesseract требуется только имя выходного файла, для которого не требуется расширение.

Работая из командной строки, это работает отлично. Однако я хотел запустить это из Java-программы и столкнулся с некоторыми ошибками.

Я нашел этот код очень полезным в качестве отправной точки

public class Main
{
   public static void main(String args[])
   {
      try
      {
         Runtime rt = Runtime.getRuntime();
         String cmdString = "cmd /c dir";

         System.out.println(cmdString);
         Process pr = rt.exec(cmdString);

         BufferedReader input = new BufferedReader(new InputStreamReader(
                                                   pr.getInputStream()));

         String line = null;

         while ((line = input.readLine()) != null)
         {
            System.out.println(line);
         }

         int exitVal = pr.waitFor();
         System.out.println("Exited with error code " + exitVal);

      }
      catch (Exception e)
      {
         System.out.println(e.toString());
         e.printStackTrace();
      }
   }
}

Распечатывает результат команды dir. Однако, когда я изменил это так

public class Main
{
   public static void main(String args[])
   {
      try
      {
         Runtime rt = Runtime.getRuntime();
         String imageFilePath = "\"C:\\Program Files (x86)\\Tesseract-OCR\\doc\\eurotext.tif\"";
         String outputFilePath = "\"C:\\Users\\Dreadnought\\Documents\\TestingFolder\\eurotext-example\"";
         String[] commands = {"cmd", "/c", "tesseract", imageFilePath, outputFilePath };

         Process pr = rt.exec(commands);

         BufferedReader input = new BufferedReader(new InputStreamReader(
               pr.getInputStream()));

         String line = null;

         while ((line = input.readLine()) != null)
         {
            System.out.println(line);
         }

         int exitVal = pr.waitFor();
         System.out.println("Exited with error code " + exitVal);
      }
      catch (Exception e)
      {
         System.out.println(e.toString());
         e.printStackTrace();
      }
   }
}

Единственное, что он выводит - это Exited with error code 1. Это ожидаемый результат, если процесс завершился с ошибкой.

Я даже попытался передать "cmd /c tesseract \"C:\\Program Files (x86)\\Tesseract-OCR\\doc\\eurotext.tif\" \"C:\\Users\\Dreadnought\\Documents\\TestingFolder\\eurotext-example\"", и у меня возникла та же ошибка.

Согласно Использование кавычек в getRuntime (). Exec я думал, что проблема заключалась в том, что я пытался избежать кавычек, поэтому я передал массив String. Но я все еще получаю Exited with error code 1.

Можно ли выполнить программу командной строки с помощью команды Java Runtime.getRuntime().exec()?


EDIT: проблема все еще возникает

Я пытался не использовать «cmd/c» мышление в соответствии с той же линией рассуждений, как предложили Евгений Дорофеев и Нандкумар Текале ниже. Однако я получаю другую ошибку:

Java.io.IOException: Cannot run program "tesseract": CreateProcess error=2, The system cannot find the file specified
Java.io.IOException: Cannot run program "tesseract": CreateProcess error=2, The system  cannot find the file specified
    at Java.lang.ProcessBuilder.start(Unknown Source)
    at Java.lang.Runtime.exec(Unknown Source)
    at Java.lang.Runtime.exec(Unknown Source)
    at Main.main(Main.Java:15)
Caused by: Java.io.IOException: CreateProcess error=2, The system cannot find the file specified
    at Java.lang.ProcessImpl.create(Native Method)
    at Java.lang.ProcessImpl.<init>(Unknown Source)
    at Java.lang.ProcessImpl.start(Unknown Source)
... 4 more

Может быть, это дает больше информации? Мне действительно любопытно, что вызывает эту проблему. Также проблема в том, добавляю ли я к предыдущим цитатам аргументы.


EDIT 2: по какой-то причине я указал абсолютный путь к исполняемому файлу тессеракта и не использовал cmd /c, работающий как талисман. Я думаю, вопрос в том, может ли Runtime.getRuntime().exec() не вызывать переменные окружения?

9
Samuel

хорошо tesseract является внешней командой, поэтому вам не нужно использовать ее с cmd. Добавьте tesseract к переменным окружения. Используйте прямую команду как:

String[] commands = {"tesseract", imageFilePath, outputFilePath };

Существующий статус 1 означает неправильную функцию. Смотрите статус выхода из процесса

1
Nandkumar Tekale

Вы не захватываете STDERR, поэтому при возникновении ошибок вы не получаете их от STDOUT (который вы захватываете). Пытаться:

BufferedReader input = new BufferedReader(new InputStreamReader(
               pr.getErrorStream()));
0
hkd93

Другой обходной путь без перекомпиляции и развертывания - использование старых путей в стиле DOS, например, C:\Program Files будет C:\Progra~1. Конечно, это будет полезно, только если вы читаете пути из файла конфигурации или БД, реестра и т.д.

0
user1382115

другой обходной путь - укажите полный путь установки файла, например /usr/local/Cellar/tesseract/3.02.02/bin/tesseract "

0
Raj Pandiri