java笔记十一:动态编译

来源:互联网 发布:学java需要先学c吗 编辑:程序博客网 时间:2024/06/06 07:35

  JDK6.0推出了动态编译的新功能,能够在程序中动态的写一个类,再对之进行编译。编译成class文件后就可以通过类加载方式把动态编译的类加载到内存中。当然也能通过RunTime类调用javac命令来动态编译。

  动态编译类的主要步骤:

  ①、写一个字符串,这个字符串就是要编译的类的全部内容。

  ②、通过输出流,把该字符串的内容写到工程下面指定的包中。

  ③、对创建的java文件进行编译。

    动态编译涉及的类有JavaCompiler、StandardJavaFileManager,具体如何编译看下面的例子。

  ④、把编译后的字节码加载到内存,然后对其进行操作。

 1 package com.compiler; 2 import java.io.*; 3 import java.lang.reflect.InvocationTargetException; 4 import java.net.URI; 5 import java.net.URL; 6 import java.net.URLClassLoader; 7 import java.util.Arrays; 8  9 import javax.tools.JavaCompiler;10 import javax.tools.SimpleJavaFileObject;11 import javax.tools.StandardJavaFileManager;12 import javax.tools.JavaFileObject; 13 import javax.tools.ToolProvider;14 15 public class CompilerAPITester {16     private static String JAVA_SOURCE_FILE = "DynamicObject.java";17     private static String JAVA_CLASS_FILE = "DynamicObject.class";18     private static String JAVA_CLASS_NAME = "DynamicObject";19 20     public static void main(String[] args) throws IOException, InterruptedException {21         //创建java文件22         String tr = "\r\n";23         String source = "package com.compiler;" + tr +24                 "public class "+JAVA_CLASS_NAME+ "{ " + tr +25                 "    public static void main(String[] args) {" + tr +26                 "        System.out.println(\"Hello World!\");" + tr +27                 "    } " + tr +28                 "}";29         String fileName = System.getProperty("user.dir")+"\\src\\com\\compiler\\"+JAVA_SOURCE_FILE;30         FileWriter fw = new FileWriter(fileName); //字符输出流31         PrintWriter pw = new PrintWriter(fw);     //将字节输出流转为PrintWriter32         pw.write(source);33         pw.close();34         //编译java文件35         JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();36         StandardJavaFileManager fileManager = compiler.getStandardFileManager(null, null, null);37         Iterable sourcefiles = fileManager.getJavaFileObjects(fileName);38         //指定编译文件存放位置,如果不指定的话,编译的文件会和java源文件在一个文件夹中39          //这样的话加载类的时候会报java.lang.ClassNotFoundException40         Iterable<String> options = Arrays.asList("-d", System.getProperty("user.dir")+"\\WebRoot\\WEB-INF\\classes");41         compiler.getTask(null, fileManager, null, options, null, sourcefiles).call();42         fileManager.close();43         //方法二:44         /*Runtime runtime = Runtime.getRuntime();45         runtime.exec("javac -d "+ System.getProperty("user.dir")+"\\WebRoot\\WEB-INF\\classes " +fileName);46         Thread.sleep(1000);*/ //因为这种方法时调用一个线程取编译,所以要让主线程睡一会儿,否则还没编译完主线程就加载类了,会导致报类无法找到的异常47         try {48             //加载类到内存49             //方法一:50             //Class c = Class.forName("com.compiler."+JAVA_CLASS_NAME);51             //方法二:52             //Class c = ClassLoader.getSystemClassLoader().loadClass("com.compiler."+JAVA_CLASS_NAME);53             //方法三:54             URL[] urls = new URL[] {new URL("file:/"+System.getProperty("user.dir")+"/src")};55             URLClassLoader loader = new URLClassLoader(urls);56             Class c = loader.loadClass("com.compiler."+JAVA_CLASS_NAME);57             //调用加载类的main方法58             c.getMethod("main",String[].class).invoke(null, (Object)new String[]{"a"});59         } catch (Exception e) {60             // TODO Auto-generated catch block61             e.printStackTrace();62         }63     }64 65 } 

 

原创粉丝点击