- 对于任意一个类,都需要由加载它的类加载器和这个类本身一同确立其在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。