java筆記十一:動態編譯


  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 } 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM