windows下Java调用可执行文件

时间:2022-12-30 08:35:37

 缘起:

  由于没有找到java转换文件的接口,因此使用java调用exe文件进行文件转换

public void convertFile(){
        Runtime rn = Runtime.getRuntime();
        Process p =null;
        try{
            p = rn.exec("D:/convert/Convert.exe D:/convert/my.ifc D:/convert/tmp.dae");
        }catch (Exception e){
            e.printStackTrace();
        }
    }

调用这个方法,就可以实现文件的转换了,和在命令窗口执行的结果一样

获取执行结果

public void convertFile(){
        Runtime rn = Runtime.getRuntime();
        Process p =null;
        try{
            p = rn.exec("D:/convert/Convert.exe D:/convert/my.ifc D:/convert/tmp.dae");int exitVal = p.waitFor();
            if (exitVal == 0) {
                System.out.println("转换成功.");
            } else {
                System.out.println( "转换失败.");
            }
        }catch (Exception e){
            e.printStackTrace();
        }finally {
            p.destroy();
        }
}

注:

  安全编码规范中都会指出:使用Process.waitfor的时候,可能导致进程阻塞,甚至死锁

  出现挂起的原因为:可执行程序的标准输出比较多,而运行窗口的标准输出缓冲区不够大。

1.主进程中调用Runtime.exec会创建一个子进程,用于执行程序。子进程创建后会和主进程分别独立运行
2.主进程需要等待脚本执行完成,然后对执行程序返回值或输出进行处理,所以这里主进程调用Process.waitfor等待子进程完成
3.子进程执行过程就是不断的打印信息。主进程中可以通过Process.getInputStream和Process.getErrorStream获取并处理
4.子进程不断向主进程发生数据,而主进程调用Process.waitfor后已挂起。当前子进程和主进程之间的缓冲区塞满后,子进程不能继续写数据,然后也会挂起
5.子进程等待主进程读取数据,主进程等待子进程结束,两个进程相互等待,最终导致死锁

  解决的办法是,利用Java中Process类提供的方法让Java虚拟机截获被调用程序的DOS运行窗口的标准输出,在waitfor()命令之前读出窗口的标准输出缓冲区中的内容。

   优化代码

public void convertFile(){
        Runtime rn = Runtime.getRuntime();
        Process p =null;
        try{
            p = rn.exec("D:/convert/Convert.exe D:/convert/my.ifc D:/convert/tmp.dae");
            InputStream stdin = p.getInputStream();
            InputStreamReader isr = new InputStreamReader(stdin);
            BufferedReader br = new BufferedReader(isr);
            String line = null;
            System.out.println("<output>");
            while ((line = br.readLine()) != null)
                System.out.println(line);
            System.out.println("</output>");
            int exitVal = p.waitFor();
            if (exitVal == 0) {
                System.out.println("转换成功.");
            } else {
                System.out.println( "转换失败.");
            }
        }catch (Exception e){
            e.printStackTrace();
        }finally {
            p.destroy();
        }
}