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); //將字節輸出流轉為PrintWriter 32 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.ClassNotFoundException 40 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 block 61 e.printStackTrace(); 62 } 63 } 64 65 }