java反射,最常用的Class.forName()方法。做畢設的時候,接收到代碼字符串,通過 JavaCompiler將代碼字符串生成A.class文件(存放在classpath下,也就是eclipse項目中的bin目錄裡),然後通過java反射機制,獲取main方法並執行。.class文件名稱固定。當 A.class文件更新的時候,問題出現了,main方法的執行結果總和第一次的執行結果相同。
代碼提交->接收代碼->編譯成A.class文件->java反射->main方法執行
具體代碼參考:http://www.cnblogs.com/hujunzheng/p/5203067.html
類加載器的委托機制!說到這裡,不得不介紹一下java的類加載器。
java虛擬機中可以安裝多個類加載器,系統默認三個主要的類加載器,每個類負責加載特定位置的類: BootStrap,ExtClassLoader,AppClassLoader
類加載器也是Java類,因為Java類的類加載器本身也是要被類加載器加載的,顯然必須有第一個類加載器不是Java類,這個正是BootStrap,使用C/C++代碼寫的,已經封裝到JVM內核中了,而ExtClassLoader和AppClassLoader是Java類。
盜圖一張:
首先我的A.class文件更新了,接著調用Class.forName()[我想的是重新加載一下字節碼文件對象],然後最終由AppClassLoader去加載,其中有一個函數很重要,就是loadClass(), 看一下這個函數的源碼,如下:
protected Class<?> loadClass(String name, boolean resolve)throws ClassNotFoundException{ //加上鎖,同步處理,因為可能是多線程在加載類 synchronized (getClassLoadingLock(name)) { //檢查,是否該類已經加載過了,如果加載過了,就不加載了 Class c = findLoadedClass(name); if (c == null) { long t0 = System.nanoTime(); try { //如果自定義的類加載器的parent不為null,就調用parent的loadClass進行加載類 if (parent != null) { c = parent.loadClass(name, false); } else { //如果自定義的類加載器的parent為null,就調用findBootstrapClass方法查找類,就是Bootstrap類加載器 c = findBootstrapClassOrNull(name); } } catch (ClassNotFoundException e) { // ClassNotFoundException thrown if class not found // from the non-null parent class loader } if (c == null) { // If still not found, then invoke findClass in order // to find the class. long t1 = System.nanoTime(); //如果parent加載類失敗,就調用自己的findClass方法進行類加載 c = findClass(name); // this is the defining class loader; record the stats sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0); sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1); sun.misc.PerfCounter.getFindClasses().increment(); } } if (resolve) { resolveClass(c); } return c; } }
如果同名的.class文件之前加載了就不會在加載了。。。
想法1: 重寫loadClass()這個函數,無論是否加載過.class問價,都重新加載。
@Override public java.lang.Class<?> loadClass(String name) throws ClassNotFoundException { System.out.println(name); byte[] data = loaderClassData(name); return this.defineClass(name, data, 0, data.length); };
但是竟然出錯了,至今還沒有搞明白... Main是我要加載的類,loadClass()函數執行了兩次,第二次不知道怎麼調用的。。。?有誰知到,告訴我一下,謝了!
Main java.lang.Object java.io.FileNotFoundException: java\lang\Object.class (系統找不到指定的路徑。) at java.io.FileInputStream.open(Native Method) at java.io.FileInputStream.<init>(Unknown Source) at com.ds.tools.MyClassLoader.loaderClassData(MyClassLoader.java:53) at com.ds.tools.MyClassLoader.loadClass(MyClassLoader.java:78) at java.lang.ClassLoader.defineClass1(Native Method) at java.lang.ClassLoader.defineClassCond(Unknown Source) at java.lang.ClassLoader.defineClass(Unknown Source) at java.lang.ClassLoader.defineClass(Unknown Source) at com.ds.tools.MyClassLoader.loadClass(MyClassLoader.java:79) at com.ds.tools.MyClassLoader.main(MyClassLoader.java:96)
想法2: 只能默默的重寫findClass()方法了, loadClass()方法中會調用這個函數,為了避過AppClassLoader檢查類是否已經加載過了,我把A.class的生成位置放到了項目根目錄下的myClass目錄中,這樣MyClassLoader委托AppClassLoader對A.class進行加載時,在當前的classpath下找不到對應的類,無法完成類的加載(同樣BootStrapLoader和ExtClassLoader都不會找到),最終是我們自定的類加載器完成類的加載,代碼如下:
public class MyClassLoader extends ClassLoader { //類加載器名稱 private String loaderName; //加載類的路徑 private String path = ""; private final String fileType = ".class"; public MyClassLoader(String loaderName){ //讓系統類加載器成為該 類加載器的父加載器 super(); this.loaderName = loaderName; } public MyClassLoader(ClassLoader parent, String loaderName){ //顯示指定該類加載器的父加載器 super(parent); this.loaderName = loaderName; } public String getPath() { return path; } public void setPath(String path) { this.path = path; } @Override public String toString() { return this.loaderName; } /** * 獲取.class文件的字節數組 * @param name * @return */ private byte[] loaderClassData(String name){ InputStream is = null; byte[] data = null; ByteArrayOutputStream baos = new ByteArrayOutputStream(); name = name.replace(".", "/"); try { is = new FileInputStream(new File(path + name + fileType)); int c = 0; while(-1 != (c = is.read())){ baos.write(c); } data = baos.toByteArray(); } catch (Exception e) { e.printStackTrace(); } finally{ try { if(is != null) is.close(); if(baos != null) baos.close(); } catch (IOException e) { e.printStackTrace(); } } return data; } /** * 獲取Class對象 */ @Override public Class<?> findClass(String name) throws ClassNotFoundException{ byte[] data = loaderClassData(name); return this.defineClass(name, data, 0, data.length); } public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException, SecurityException, NoSuchMethodException, IllegalArgumentException, InvocationTargetException { for(int i=0; i<5; i++){ MyClassLoader loader1 = new MyClassLoader("MyClassLoader"); //String path = new File(MyClassLoader.getSystemClassLoader().getResource("").getPath()).getParent(); loader1.setPath("myClass/"); Class<?> clazz = loader1.loadClass("Main"); System.out.println(clazz.getName()); } } }