如何在Java程序中運(yùn)行外部類文件
一、引言
無(wú)論是用傳統(tǒng)的編程語(yǔ)言(C++、VB等)還是Java語(yǔ)言編程,都經(jīng)常需要在一個(gè)運(yùn)行的程序中執(zhí)行另外一個(gè)獨(dú)立的外部程序。例如用Java設(shè)計(jì)一個(gè)IDE程序,那么這個(gè)IDE程序就必需能夠調(diào)式、運(yùn)行其它獨(dú)立的外部Java程序。況且直接運(yùn)行已經(jīng)存在的外部程序來(lái)實(shí)現(xiàn)本程序的某些特定的功能,也是提高程序開發(fā)效率的一種重要手段。
Java2為實(shí)現(xiàn)在一個(gè)Java程序中運(yùn)行外部類文件(即Java程序)提供了的兩種解決方案,即在同一進(jìn)程中運(yùn)行外部類文件和在不同進(jìn)程中運(yùn)行外部類文件。
二、在同一進(jìn)程中運(yùn)行外部類文件
Java規(guī)定任何Java應(yīng)用(application)的入口都是:main(),因此要實(shí)現(xiàn)在同一進(jìn)程中運(yùn)行外部類文件,只要想辦法在程序中直接調(diào)用外部類文件的main()方法即可。
Java2中的映象(Reflection)API可以對(duì)Java虛擬機(jī)中的類、接口、對(duì)象等進(jìn)行映象,通過(guò)它能夠動(dòng)態(tài)地得到類、接口、對(duì)象的各種信息,還能夠動(dòng)態(tài)地設(shè)置對(duì)象的域的值,并可以動(dòng)態(tài)地調(diào)用對(duì)象中的各種方法。
因此,我們可以利用Reflection API在運(yùn)行時(shí)(runtime)動(dòng)態(tài)地確定外部類文件的main()方法,然后調(diào)用它。例如:
假定將被執(zhí)行的外部類文件(Invoked.java)如下:
這個(gè)例子簡(jiǎn)單展示了如何在同一進(jìn)程(process)中執(zhí)行外部類文件,不僅如此,而且還是在同一線程(thread)中,當(dāng)然完全可以在不同的線程中來(lái)完成。
三、在不同進(jìn)程中運(yùn)行外部類文件
要在不同進(jìn)程中運(yùn)行外部類文件,只要想辦法在程序中直接執(zhí)行:
java 外部類文件 (本例即為:java Invoked)即可。
幸運(yùn)的是java.lang.Runtime 類提供了exce()方法來(lái)完成對(duì)外部可執(zhí)行程序的調(diào)用。注意:在這里exce()方法調(diào)用的其實(shí)是java解釋器這個(gè)可執(zhí)行程序,外部類文件Invoked.class只不過(guò)是作為java解釋器的參數(shù)出現(xiàn)。事實(shí)上,exce()方法還可以直接調(diào)用其它的可執(zhí)行文件,而不僅僅是java解釋器。
假定將被執(zhí)行的外部類文件仍然是上面的Invoked.java;
那么運(yùn)行上述類文件的主程序(InvokerFromSeparateProcess.java)如下:
你也許會(huì)發(fā)現(xiàn)并沒(méi)有看到被調(diào)用的外部類文件Invoked.class的輸出。問(wèn)題在于新進(jìn)程(new porcess)的標(biāo)準(zhǔn)輸出流(standard output stream)并不會(huì)自動(dòng)地定向到stdout。為了得到并顯示新進(jìn)程的輸出信息,我們可以利用p.getInputStream()來(lái)讀取新進(jìn)程的輸出信息,然后把讀到的信息發(fā)送到標(biāo)準(zhǔn)輸出流(standard output)上。
四、結(jié)語(yǔ)
利用Java2中的映象(Reflection)API實(shí)現(xiàn)在同一進(jìn)程(process)中執(zhí)行外部類文件的方法,優(yōu)點(diǎn)是消耗資源較少,因?yàn)樗淮蜷_新的進(jìn)程;功能強(qiáng)大,因?yàn)樗坏梢詧?zhí)行外部類文件的main()方法,而且還可以執(zhí)行外部類文件的其它方法,可以非常細(xì)致地利用外部類文件的一切資源。缺點(diǎn)是它只能執(zhí)行用Java編寫的外部類文件,而不能執(zhí)行其它形式的可執(zhí)行文件,如EXE、COM等;其次是編程較第二種形式稍微復(fù)雜一點(diǎn)。
利用java.lang.Runtime 類提供了exce()方法來(lái)完成對(duì)外部可執(zhí)行程序的調(diào)用的方法,優(yōu)點(diǎn)是編程簡(jiǎn)單,并且可以運(yùn)行一切可執(zhí)行程序,包括用其它語(yǔ)言編寫的程序。缺點(diǎn)是消耗資源較多,因?yàn)樗蜷_新的進(jìn)程;其次是它只能整體運(yùn)行外部類文件,而不能細(xì)致地訪問(wèn)被調(diào)用程序的內(nèi)部資源。
一般情況下,如果要運(yùn)行用Java編寫的外部類文件,選擇第一種方法較為靈活、恰當(dāng);如果要運(yùn)行用其它語(yǔ)言編寫的可執(zhí)行程序,選擇第二種方法較為方便、直接,但是一般情況下要盡量避免這樣做,否則將降低Java程序的跨平臺(tái)性,因?yàn)檫@種外部程序一般是平臺(tái)相關(guān)的。
無(wú)論是用傳統(tǒng)的編程語(yǔ)言(C++、VB等)還是Java語(yǔ)言編程,都經(jīng)常需要在一個(gè)運(yùn)行的程序中執(zhí)行另外一個(gè)獨(dú)立的外部程序。例如用Java設(shè)計(jì)一個(gè)IDE程序,那么這個(gè)IDE程序就必需能夠調(diào)式、運(yùn)行其它獨(dú)立的外部Java程序。況且直接運(yùn)行已經(jīng)存在的外部程序來(lái)實(shí)現(xiàn)本程序的某些特定的功能,也是提高程序開發(fā)效率的一種重要手段。
Java2為實(shí)現(xiàn)在一個(gè)Java程序中運(yùn)行外部類文件(即Java程序)提供了的兩種解決方案,即在同一進(jìn)程中運(yùn)行外部類文件和在不同進(jìn)程中運(yùn)行外部類文件。
二、在同一進(jìn)程中運(yùn)行外部類文件
Java規(guī)定任何Java應(yīng)用(application)的入口都是:main(),因此要實(shí)現(xiàn)在同一進(jìn)程中運(yùn)行外部類文件,只要想辦法在程序中直接調(diào)用外部類文件的main()方法即可。
Java2中的映象(Reflection)API可以對(duì)Java虛擬機(jī)中的類、接口、對(duì)象等進(jìn)行映象,通過(guò)它能夠動(dòng)態(tài)地得到類、接口、對(duì)象的各種信息,還能夠動(dòng)態(tài)地設(shè)置對(duì)象的域的值,并可以動(dòng)態(tài)地調(diào)用對(duì)象中的各種方法。
因此,我們可以利用Reflection API在運(yùn)行時(shí)(runtime)動(dòng)態(tài)地確定外部類文件的main()方法,然后調(diào)用它。例如:
假定將被執(zhí)行的外部類文件(Invoked.java)如下:
public class Invoked {
public static void main(String[] args) {
if (args.length != 2) {
System.err.println("Usage: java Invoked name sex ");
System.exit(1);
}
System.out.println("Hello, I come from OutClassFile!");
System.out.println("my name is "+args[0]);
System.out.println("my sex is "+args[1]);
}
}
運(yùn)行上述類文件的主程序(Invoker.java)一般具有如下形式:
import java.lang.reflect.*;
public class Invoker {
public static void main(String[] args) {
//args[0]存儲(chǔ)被執(zhí)行的外部類名,本例中args[0]=“Invoked”
if (args.length != 1) {
System.err.println("Usage: java Invoker ");
System.exit(1);
}
Class[] argTypes = new Class[1];
argTypes[0] = String[].class;
try {
Method mainMethod = Class.forName(args[0]).getDeclaredMethod("main",argTypes);
Object[] argListForInvokedMain = new Object[1];
//被調(diào)用的main()方法的參數(shù)個(gè)數(shù)=1
argListForInvokedMain[0] = new String[2];
//被調(diào)用的main()方法的參數(shù)是一個(gè)長(zhǎng)度為2的String數(shù)組
String argsToinvoked[]={"WangShuangLin","Man"};
ArgListForInvokedMain[0]= argsToinvoked;
//設(shè)置傳遞給被調(diào)用的外部類的main()方法的參數(shù)
mainMethod.invoke(null, argListForInvokedMain);
//inkoke(Object obj, Object[] args)是調(diào)用方法;
//其中obj是方法所在的對(duì)象實(shí)例; 但因?yàn)檫@里的main()是靜態(tài)方法,
//所以無(wú)須實(shí)例化,填上null即可;
//args是傳遞給該方法的參數(shù)。
}
catch (ClassNotFoundException ex) {
System.err.println("Class"+args[0]+"not found in classpath.");
}
catch (NoSuchMethodException ex) {
System.err.println("Class "+args[0]+" does not define public static void main(String[])");
}
catch (InvocationTargetException ex) {
System.err.println("Exception while executing "+args[0]+ ":"+ex.getTargetException());
}
catch (IllegalAccessException ex) {
System.err.println("main(String[]) in class "+args[0] + "is not public");
}
}
}
用如下的命令行運(yùn)行主程序:java Invoker Invoked。 這個(gè)例子簡(jiǎn)單展示了如何在同一進(jìn)程(process)中執(zhí)行外部類文件,不僅如此,而且還是在同一線程(thread)中,當(dāng)然完全可以在不同的線程中來(lái)完成。
三、在不同進(jìn)程中運(yùn)行外部類文件
要在不同進(jìn)程中運(yùn)行外部類文件,只要想辦法在程序中直接執(zhí)行:
java 外部類文件 (本例即為:java Invoked)即可。
幸運(yùn)的是java.lang.Runtime 類提供了exce()方法來(lái)完成對(duì)外部可執(zhí)行程序的調(diào)用。注意:在這里exce()方法調(diào)用的其實(shí)是java解釋器這個(gè)可執(zhí)行程序,外部類文件Invoked.class只不過(guò)是作為java解釋器的參數(shù)出現(xiàn)。事實(shí)上,exce()方法還可以直接調(diào)用其它的可執(zhí)行文件,而不僅僅是java解釋器。
假定將被執(zhí)行的外部類文件仍然是上面的Invoked.java;
那么運(yùn)行上述類文件的主程序(InvokerFromSeparateProcess.java)如下:
public class InvokerFromSeparateProcess {
public static void main(String[] args) {
if (args.length != 1) {
System.err.println("Usage: java InvokerFromSeparateProcess ");
System.exit(1);
}
String name=” WangShuangLin”;
String sex=” man”;
try {
Process p = Runtime.getRuntime().exec("java "+args[0]+name+sex);
}
catch (java.io.IOException ex) {
System.err.println("Problems invoking class "+args[0]+": "+ex);
}
}
}
用如下的命令行運(yùn)行主程序:java InvokerFromSeparateProcess Invoked。 你也許會(huì)發(fā)現(xiàn)并沒(méi)有看到被調(diào)用的外部類文件Invoked.class的輸出。問(wèn)題在于新進(jìn)程(new porcess)的標(biāo)準(zhǔn)輸出流(standard output stream)并不會(huì)自動(dòng)地定向到stdout。為了得到并顯示新進(jìn)程的輸出信息,我們可以利用p.getInputStream()來(lái)讀取新進(jìn)程的輸出信息,然后把讀到的信息發(fā)送到標(biāo)準(zhǔn)輸出流(standard output)上。
四、結(jié)語(yǔ)
利用Java2中的映象(Reflection)API實(shí)現(xiàn)在同一進(jìn)程(process)中執(zhí)行外部類文件的方法,優(yōu)點(diǎn)是消耗資源較少,因?yàn)樗淮蜷_新的進(jìn)程;功能強(qiáng)大,因?yàn)樗坏梢詧?zhí)行外部類文件的main()方法,而且還可以執(zhí)行外部類文件的其它方法,可以非常細(xì)致地利用外部類文件的一切資源。缺點(diǎn)是它只能執(zhí)行用Java編寫的外部類文件,而不能執(zhí)行其它形式的可執(zhí)行文件,如EXE、COM等;其次是編程較第二種形式稍微復(fù)雜一點(diǎn)。
利用java.lang.Runtime 類提供了exce()方法來(lái)完成對(duì)外部可執(zhí)行程序的調(diào)用的方法,優(yōu)點(diǎn)是編程簡(jiǎn)單,并且可以運(yùn)行一切可執(zhí)行程序,包括用其它語(yǔ)言編寫的程序。缺點(diǎn)是消耗資源較多,因?yàn)樗蜷_新的進(jìn)程;其次是它只能整體運(yùn)行外部類文件,而不能細(xì)致地訪問(wèn)被調(diào)用程序的內(nèi)部資源。
一般情況下,如果要運(yùn)行用Java編寫的外部類文件,選擇第一種方法較為靈活、恰當(dāng);如果要運(yùn)行用其它語(yǔ)言編寫的可執(zhí)行程序,選擇第二種方法較為方便、直接,但是一般情況下要盡量避免這樣做,否則將降低Java程序的跨平臺(tái)性,因?yàn)檫@種外部程序一般是平臺(tái)相關(guān)的。