- 對于任意一個類,都需要由加載它的類加載器和這個類本身一同確立其在Java虛擬機中的唯一性,每一個類加載器,都擁有一個獨立的類名稱空間。
- 上面話可以表達得更通俗一些:比較兩個類是否“相等”,只有在這兩個類是由同一個類加載器加載的前提下才有意義,否則,即使這兩個類來源于同一個Class文件,被同一個虛擬機加載,只要加載它們的類加載器不同,那這兩個類就必定不相等。這里所指的“相等”,包括代表類的Class對象的equals()方法、isAssignableFrom()方法、isInstance()方法的返回結果,也包括使用instanceof關鍵字做對象所屬關系判定等情況。如果沒有注意到類加載器的影響,在某些情況下可能會產生具有迷惑性的結果,下面演示了不同的類加載器對instanceof關鍵字運算的結果的影響。
package com.zlcook.action;
import java.io.IOException;
import java.io.InputStream;
/**
* @author zlcook
* @version 創建時間:2017年4月12日 下午8:08:22
* 類加載器與instanceof關鍵字演示
*/
public class MyClassLoaderTest {
public static void main(String[] args) throws Exception {
ClassLoader myLoader = new ClassLoader() {
@Override
public Class<?> loadClass(String name) throws ClassNotFoundException {
try {
String fileName=name.substring(name.lastIndexOf(".")+1)+".class";
InputStream is=getClass().getResourceAsStream(fileName);
if( is == null ){
return super.loadClass(name);
}
byte[] bytes = new byte[is.available()];
is.read(bytes); //通過自定義類加載器讀取class文件的二進制流
return defineClass(name, bytes, 0,bytes.length);
} catch (IOException e) {
e.printStackTrace();
throw new ClassNotFoundException(name);
}
}
};
Object obj = myLoader.loadClass("com.zlcook.action.MyClassLoaderTest").newInstance();
System.out.println(obj.getClass() );
System.out.println(obj instanceof MyClassLoaderTest);
}
}
運行結果:
class com.zlcook.action.MyClassLoaderTest
false
- 自定義的類加載器myLoader可以加載與自己在同一路徑下的Class文件。我們使用這個類加載器去加載了一個名為“com.zlcook.action.MyClassLoaderTest”的類,并實例化了這個類的對象。
- 兩行輸出結果中,從第一句可以看出,這個對象確實是類com.zlcook.action.MyClassLoaderTest實例化出來的對象,但從第二句可以發現,這個對象與類com.zlcook.action.MyClassLoaderTest做所屬類型檢查的時候卻返回了false,這是因為虛擬機中存在了兩個MyClassLoaderTest類,一個是由系統應用程序類加載器加載的,另外一個是由我們自定義的類加載器加載的,雖然都來自同一個Class文件,但依然是兩個獨立的類,做對象所屬類型檢查時結果自然為false。