前言
最近遇到了一个问题:由不同类加载器加载同一个类,实例化为对象。使用instanceof
判断该对象与该类的归属,请问结果是true
还是false
?
答案是false
。
解析
验证
import java.io.IOException;
import java.io.InputStream;
public class ClassLoaderTest {
public static void main(String[] args) throws Exception {
ClassLoader myLoader = new ClassLoader() {
@SuppressWarnings("ResultOfMethodCallIgnored")
@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[] b = new byte[is.available()];
is.read(b);
return defineClass(name, b, 0, b.length);
} catch (IOException e) {
throw new ClassNotFoundException(name);
}
}
};
Object obj = myLoader.loadClass("Practice.Java.ClassLoaderTest").newInstance();
System.out.println(obj.getClass());
System.out.println(obj instanceof Practice.Java.ClassLoaderTest);
}
}
- 输出结果:
class Practice.Java.ClassLoaderTest
false
原因
《深入理解JVM虚拟机:JVM高级特性与最佳实践》
对于任意一个类,都需要由加载它的类加载器和这个类本身一同确立其在Java虚拟机中的唯一性,每一个类加载器,都拥有一个独立的类名称空间。这句话可以表达得更通俗一些:比较两个类是否“相等”,只有在这两个类是由同一个类加载器加载的前提下才有意义,否则,即使这两个类来源于同一个Class文件,被同一个虚拟机加载,只要加载它们的类加载器不同,那这两个类就必定不相等。
结果分析
《深入理解JVM虚拟机:JVM高级特性与最佳实践》
两行输出结果中,从第一句可以看出,这个对象确实是类Practice.Java.ClassLoaderTest实例化出来的对象,但从第二句可以发现,这个对象与类Practice.Java.ClassLoaderTest做所属类型检查的时候却返回了false。
这是因为虚拟机中存在了两个ClassLoaderTest类,一个是由系统应用程序类加载器加载的,另外一个是由我们自定义的类加载器加载的。虽然都来自同一个Class文件,但依然是两个独立的类,做对象所属类型检查时结果自然为false。
结论
Java类加载器这种特性可以简单的总结为命名空间。即在 Java 虚拟机中,类的唯一性是由类加载器实例以及类的全名一同确定的。即便是同一串字节流,经由不同的类加载器加载,也会得到两个不同的类。
附录
- 《深入理解JVM虚拟机:JVM高级特性与最佳实践》