2012-03-06 20:36:06|?次阅读|上传:wustguangh【已有?条评论】发表评论
关键词:Java, 操作系统|来源:唯设编程网
在一个java应用中,经常会遇到这样的需求,就是需要调用一些外部的应用做一些处理,比如调用excel,然后再继续程序的运行。本文总结了Java启动外部程序的常用方式,包括启动DOS内部命令、打开系统关联文件、执行DOS应用程序等,让java应用更加灵活。
首先介绍启动外部程序的基本方法。
Runtime.getRuntime().exec("notepad.exe");
记事本被打开了是吧。
Runtime runtime=Runtime.getRuntime(); String[] commandArgs={"notepad.exe","c:/boot.ini"}; runtime.exec(commandArgs);
现在不单单打开了记事本,而且还装载了boot.ini文件。
如果要执行一条DOS内部命令,有两种方法。一种方法是把命令解释器包含在exec()的参数中。例如,执行dir命令,在NT上,可写成exec("cmd.exe /c dir"),在windows95/98下,可写成“command.exe /c dir”,其中参数“/c”表示命令执行后关闭DOS立即关闭窗口。另一种方法是,把内部命令放在一个批命令my_dir.bat文件中,在Java程序中写成exec("my_dir.bat")。如果仅仅写成exec("dir"),Java虚拟机则会报运行时错误。前一种方法要保证程序的可移植性,需要在程序中读取运行的操作系统平台,以调用不同的命令解释器。后一种方法则不需要做更多的处理。
打开一个不可执行的文件,但该文件存在关联的应用程序,则可以有两种方式。以打开一个word文档a.doc文件为例,Java中可以有以下两种写法:
exec("start .a.doc"); exec("FilesMicrosoft Officeofficewinword.exe .a.doc");
显然,前一种方法更为简捷方便。
在Windows平台上,运行被调用程序的DOS窗口在程序执行完毕后往往并不会自动关闭,从而导致Java应用程序阻塞在waitfor()语句。导致该现象的一个可能的原因是,该可执行程序的标准输出比较多,而运行窗口的标准输出缓冲区不够大。解决的办法是,利用Java中Process类提供的方法让Java虚拟机截获被调用程序的DOS运行窗口的标准输出,在waitfor()命令之前读出窗口的标准输出缓冲区中的内容。一段典型的程序如下:
String s; Process process = Runtime.getRuntime().exec("cmd /c dir windows"); BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(process.getInputStream()); while((s=bufferedReader.readLine()) != null) System.out.println(s); process.waitfor();
现在已经完全解决了调用外部程序的问题,我们接着讨论一下exec函数的返回值,Runtime实例的exec函数返回一个新的 Process 对象,用于管理子进程,下面对Process对象的一些重要方法进行讨论并给出实际代码。
导致当前线程等待,如有必要,一直要等到由该 Process
对象表示的进程已经终止。如果已终止该子进程,此方法立即返回。如果没有终止该子进程,调用的线程将被阻塞,直到退出子进程。
函数原型:
public abstract int waitFor() throws InterruptedException
返回值:
进程的出口值。根据惯例,0 表示正常终止。
实例:
Runtime runtime=Runtime.getRuntime(); String[] commandArgs={"notepad.exe","c:/boot.ini"}; Process process=runtime.exec(commandArgs); int exitcode=process.waitFor(); System.out.println("finish:"+exitcode);
执行上面的代码以后发现不同的地方了吗,waitFor会使线程阻塞,只有外部程序退出后才会执行System.out.println("finish:"+exitcode);
这个功能很有用是吧,因为多数时候你都需要等待用户处理完外部程序以后才继续你的java应用。
杀掉子进程。强制终止此 Process
对象表示的子进程。
public void RuntimeTest() throws IOException, InterruptedException { Runtime runtime = Runtime.getRuntime(); String[] commandArgs = { "notepad.exe", "c:/boot.ini" }; final Process process = runtime.exec(commandArgs); new Thread(new Runnable() { @Override public void run() { try { Thread.sleep(5000); } catch (InterruptedException e) { } process.destroy(); } }).start(); int exitcode = process.waitFor(); System.out.println("finish:" + exitcode); }
这个演示稍微复杂了一些,如果你等待5秒,就会发现记事本自动关闭了,是的,这个就是destroy方法的作用,强制关闭调用的外部程序。
不用我解释了吧,这是非常有用的方法。
以上的部分已经足够你调用并控制你的外部应用了。如果需要更详细的信息,看javadoc文档吧。
最后的说明:ProcessBuilder这个1.5新增的类也可以完成同样的任务,Runtime就是调用了这个类。