JDK6的新特性之四:使用CompilerAPI
现在我们可以用JDK6的CompilerAPI(JSR199)去动态编译Java源文件,CompilerAPI结合反射功能就可以实现动态的产生Java代码并编译执行这些代码,有点动态语言的特征。这个特性对于某些需要用到动态编译的应用程序相当有用,比如JSPWebServer,当我们手动修改JSP后,是不希望需要重启WebServer才可以看到效果的,这时候我们就可以用CompilerAPI来实现动态编译JSP文件,当然,现在的JSPWebServer也是支持JSP热部署的,现在的JSPWebServer通过在运行期间通过Runtime.exec或ProcessBuilder来调用javac来编译代码,这种方式需要我们产生另一个进程去做编译工作,不够优雅而且容易使代码依赖与特定的操作系统;CompilerAPI通过一套易用的标准的API提供了更加丰富的方式去做动态编译,而且是跨平台的。下面代码演示了CompilerAPI的使用
publicclassCompilerAPITester{
privatestaticStringJAVA_SOURCE_FILE="DynamicObject.java";
privatestaticStringJAVA_CLASS_FILE="DynamicObject.class";
privatestaticStringJAVA_CLASS_NAME="DynamicObject";
publicstaticvoidmain(String[]args){
JavaCompilercompiler=ToolProvider.getSystemJavaCompiler();
StandardJavaFileManagerfileManager=compiler.getStandardFileManager(null,null,null);
generateJavaClass();
try{
//将产生的类文件拷贝到程序的ClassPath下面,下面这一行代码是特定于Windows+IntelliJIDEA6.0项目,不具有移植性
Runtime.getRuntime().exec("cmd/ccopy"+JAVA_CLASS_FILE+"classes//production//JDK6Features");
Iterable<?extendsJavaFileObject>sourcefiles=fileManager.getJavaFileObjects(JAVA_SOURCE_FILE);
compiler.getTask(null,fileManager,null,null,null,sourcefiles).call();
fileManager.close();
Class.forName(JAVA_CLASS_NAME).newInstance();//创建动态编译得到的DynamicObject类的实例
}catch(Exceptionex){
ex.printStackTrace();
}
}
publicstaticvoidgenerateJavaClass(){
try{
FileWriterfw=newFileWriter(JAVA_SOURCE_FILE);
BufferedWriterbw=newBufferedWriter(fw);
bw.write("publicclass"+JAVA_CLASS_NAME+"{");
bw.newLine();
bw.write("public"+JAVA_CLASS_NAME+"(){System.out.println(/"IntheconstructorofDynamicObject/");}}");
bw.flush();
bw.close();
}catch(IOExceptionex){
ex.printStackTrace();
}
}
}
程序运行后,会产生DynamicObject.java和DynamicObject.class两个文件,并在控制台输出
IntheconstructorofDynamicObject
现在我们可以用JDK6的CompilerAPI(JSR199)去动态编译Java源文件,CompilerAPI结合反射功能就可以实现动态的产生Java代码并编译执行这些代码,有点动态语言的特征。这个特性对于某些需要用到动态编译的应用程序相当有用,比如JSPWebServer,当我们手动修改JSP后,是不希望需要重启WebServer才可以看到效果的,这时候我们就可以用CompilerAPI来实现动态编译JSP文件,当然,现在的JSPWebServer也是支持JSP热部署的,现在的JSPWebServer通过在运行期间通过Runtime.exec或ProcessBuilder来调用javac来编译代码,这种方式需要我们产生另一个进程去做编译工作,不够优雅而且容易使代码依赖与特定的操作系统;CompilerAPI通过一套易用的标准的API提供了更加丰富的方式去做动态编译,而且是跨平台的。下面代码演示了CompilerAPI的使用
publicclassCompilerAPITester{
privatestaticStringJAVA_SOURCE_FILE="DynamicObject.java";
privatestaticStringJAVA_CLASS_FILE="DynamicObject.class";
privatestaticStringJAVA_CLASS_NAME="DynamicObject";
publicstaticvoidmain(String[]args){
JavaCompilercompiler=ToolProvider.getSystemJavaCompiler();
StandardJavaFileManagerfileManager=compiler.getStandardFileManager(null,null,null);
generateJavaClass();
try{
//将产生的类文件拷贝到程序的ClassPath下面,下面这一行代码是特定于Windows+IntelliJIDEA6.0项目,不具有移植性
Runtime.getRuntime().exec("cmd/ccopy"+JAVA_CLASS_FILE+"classes//production//JDK6Features");
Iterable<?extendsJavaFileObject>sourcefiles=fileManager.getJavaFileObjects(JAVA_SOURCE_FILE);
compiler.getTask(null,fileManager,null,null,null,sourcefiles).call();
fileManager.close();
Class.forName(JAVA_CLASS_NAME).newInstance();//创建动态编译得到的DynamicObject类的实例
}catch(Exceptionex){
ex.printStackTrace();
}
}
publicstaticvoidgenerateJavaClass(){
try{
FileWriterfw=newFileWriter(JAVA_SOURCE_FILE);
BufferedWriterbw=newBufferedWriter(fw);
bw.write("publicclass"+JAVA_CLASS_NAME+"{");
bw.newLine();
bw.write("public"+JAVA_CLASS_NAME+"(){System.out.println(/"IntheconstructorofDynamicObject/");}}");
bw.flush();
bw.close();
}catch(IOExceptionex){
ex.printStackTrace();
}
}
}
程序运行后,会产生DynamicObject.java和DynamicObject.class两个文件,并在控制台输出
IntheconstructorofDynamicObject