为什么命令行实用程序在通过java程序调用时会挂起?

时间:2012-05-09 11:29:34

标签: java linux barcode-scanner runtime.exec processbuilder

我需要你在以下任务中提出建议和指导。 我使用libdmtx,它附带一个命令行实用程序,它读取ECC200 Data Matrix条形码的图像文件,读取它们的内容,并将解码的消息写入标准输出。 我想在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类的新代码,这段代码也提供了与上面代码相​​同的输出,它挂在了这一行 处理过程= 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);
}

}

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

3 个答案:

答案 0 :(得分:4)

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

为简单起见,我建议您使用ProcessBuilder代替Runtime.exec(),这样可以合并两个InputStream,如下所示:

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

所以,现在你可以只读一个。

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

希望有所帮助

答案 1 :(得分:2)

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

  1. 如果程序没有向stderr打印任何内容,您将挂在第2行。
  2. 如果程序向stderr发送太多内容以填充缓冲区,则程序本身将阻塞,Java将在waitFor阻止。
  3. 这两个都适用于stdout。
  4. 在您链接的文章中详细介绍了使用流程输出流的正确方法。接受这个建议,没有人可以给你更好的建议。

答案 2 :(得分:0)

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

想象一下,该程序想要输出2行文本。或者只有一行而是进入stderr。您的代码只从stdout读取 1行,然后等待进程退出。这意味着子程序可能会等待读者读取下一行,因此它会在write中等待,直到有人解锁管道 - 永远。

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