3

我需要您在以下任务中的建议和指导。我正在使用带有命令行实用程序的 libdmtx,它读取 ECC200 数据矩阵条形码的图像文件,读取它们的内容,并将解码的消息写入标准输出。我想在 linux 平台上的 java 程序中使用这个命令行实用程序。我正在使用 ubuntu linux。我已经在我的 linux 机器上安装了 libdmtx。当我调用命令时

dmtxread -n /home/admin/ab.tif

在 linux 终端上,它立即给出图像中条形码的解码值。

当我要使用我的 java 程序调用此命令时,代码会在执行命令时出现问题,并且 dotn 会给出输出。看起来程序正在处理或挂起。

以下是我的 java 代码,它调用以下命令

import java.io.BufferedReader;
import java.io.File;
import java.io.IOException;
import java.io.InputStreamReader;


public class Classtest {

    public static void getCodes(){

        try 
        { 
            Process p; 
            String command[]=new String[3];
            command[0]="dmtxread";
            command[1]="-n";
            command[2]="/home/admin/ab.tif";

            System.out.println("Command : "+command[0]+command[1]+command[2]);
            p=Runtime.getRuntime().exec(command);  //I think hangs over here.

            BufferedReader reader=new BufferedReader(new InputStreamReader(p.getErrorStream()));
            String line=reader.readLine();
            if(line==null){
                reader=new BufferedReader(new InputStreamReader(p.getInputStream()));
                line=reader.readLine();
                System.out.print("Decoded      :- "+line);
            }else{
                System.out.print("Error      :- "+line);
            }
            System.out.println(p.waitFor());

        }catch(IOException e1) {
            e1.getMessage();
            e1.printStackTrace();
        }catch(InterruptedException e2) {
            e2.getMessage();
            e2.printStackTrace();
        } 


    }
    public static void main(String args[]){
        getCodes();
    }

}

请告诉我的朋友我的代码哪里出错了。

我参考了以下文章,但没有得到任何帮助

http://www.javaworld.com/javaworld/jw-12-2000/jw-1229-traps.html?page=1

请朋友们指导一下!谢谢!

这是我使用 ProcessBuilder 类的新代码,该代码也提供与上述代码相同的输出,即它挂在 Process process = pb.start();

public class Test {

public static void main(final String[] args) throws IOException, InterruptedException {
    //Build command 
    List<String> commands = new ArrayList<String>();
    commands.add("dmtxread");
    commands.add("-n");
    commands.add("/home/admin/ab.tif");
    System.out.println(commands);

    //Run macro on target
    ProcessBuilder pb = new ProcessBuilder(commands);
    pb.redirectErrorStream(true);
    Process process = pb.start();

    //Read output
    StringBuilder out = new StringBuilder();
    BufferedReader br = new BufferedReader(new InputStreamReader(process.getInputStream()));
    String line = null, previous = null;
    while ((line = br.readLine()) != null){
        System.out.println(line);
    }

    //Check result
    if (process.waitFor() == 0)
        System.out.println("Success!");
    System.exit(0);

    //Abnormal termination: Log command parameters and output and throw        ExecutionException
    System.err.println(commands);
    System.err.println(out.toString());
    System.exit(1);
}

}

请指导我解决这个问题。感谢您!

4

3 回答 3

4

readLine 会阻塞,直到它从错误流中接收到新行。因此,如果没有输出,您的程序将无法通过第一个 readLine。

为简单起见,我建议您使用 aProcessBuilder而不是Runtime.exec(),它可以让您合并两个 InputStream,如下所示:

ProcessBuilder builder = new ProcessBuilder(cmd,arg0,arg1);
builder.redirectErrorStream(true);
Process process = builder.start();

所以,现在你可以从一个中读取。

或者,您可以使用单独的线程来使用两个 InputStream。

希望有帮助

于 2012-05-09T11:47:31.957 回答
2

您的流消费代码非常混乱。您尝试从 stderr 读取一行,然后放弃该读取器,然后尝试从 stdout 读取一行。

  1. 如果程序没有向 stderr 打印任何内容,您将在第 2 行挂起。
  2. 如果程序向 stderr 发送了太多东西以至于它填满了它的缓冲区,那么程序本身就会阻塞,而你的 Java 会阻塞在waitFor.
  3. 这两个都适用于标准输出。

您链接的那篇文章详细介绍了使用流程输出流的正确方法。接受这个建议,没有人能给你比这更好的建议。

于 2012-05-09T11:42:08.577 回答
0

我不确定您的程序到底发生了什么以及它挂在哪里(您可以使用调试器或跟踪输出来检查),但这是可能的情况:

假设程序要输出 2 行文本。或者只有一行但进入标准错误。您的代码仅从标准输出读取1 行,然后等待进程退出。这意味着子程序可能会等待读者阅读下一行,因此它会一直等待,write直到有人解除管道阻塞——永远。

当你从命令行运行dmtxread时,输出管道上没有阻塞,所以程序运行得很好。

于 2012-05-09T11:37:29.543 回答