我需要你在以下任务中提出建议和指导。 我使用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);
}
}
请指导我解决这个问题。 谢谢你!
答案 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读取一行。
waitFor
阻止。在您链接的文章中详细介绍了使用流程输出流的正确方法。接受这个建议,没有人可以给你更好的建议。
答案 2 :(得分:0)
我不确定你的程序究竟发生了什么,它在哪里挂起(你可以使用调试器或跟踪输出来检查),但这是可能的情况:
想象一下,该程序想要输出2行文本。或者只有一行而是进入stderr。您的代码只从stdout读取 1行,然后等待进程退出。这意味着子程序可能会等待读者读取下一行,因此它会在write
中等待,直到有人解锁管道 - 永远。
当您从命令行运行dmtxread
时,输出管道上没有阻塞,因此程序运行得很好。