类加载器通过一个类的全限定名来获取描述此类的二进制字节流。
类加载器在类层次划分、OSGi、热部署、代码加密等领域发挥着重要的作用。
比较两个类是否“相等”,只有在这两个类是由同一个类加载器加载的前提之下才有意义,否则,即使这两个类是来源于同一个 Class 文件,只要加载它们的类加载器不同,那这两个类就必定不相等。这里的“相等”包括 equal() 方法、isAssignableForm() 方法、isInstance() 方法和 instanceof 关键字。
下面的例子可以看到,虽然都是来自同一个 Class 文件,但是因为类加载器不同,依然是两个独立的类,自然不会“相等”。
public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException {
// 自定义简单类加载器
ClassLoader myClassLoader = new ClassLoader() {
@Override
public Class<?> loadClass(String name) throws ClassNotFoundException {
try {
String fileName = name.substring(name.lastIndexOf(".") + 1) + ".class";
InputStream inputStream = getClass().getResourceAsStream(fileName);
if (inputStream == null) {
return super.loadClass(name);
}
byte[] bytes = new byte[inputStream.available()];
inputStream.read(bytes);
return defineClass(name, bytes, 0, bytes.length);
} catch (IOException e) {
e.printStackTrace();
}
return super.loadClass(name);
}
};
Object newInstance = myClassLoader.loadClass("org.jvm.demo.chapter7.ClassLoaderTest").newInstance();
System.out.println(newInstance.getClass()); // org.jvm.demo.chapter7.ClassLoaderTest
System.out.println(newInstance instanceof org.jvm.demo.chapter7.ClassLoaderTest); // false
}
绝大部分 Java 程序都会使用到以下三种系统提供的类加载器:
如图所示类加载器之间的层次关系,就称为类加载器的双亲委派模型。双亲委派模型要求除了顶层的启动类加载器外,其余的类加载器都应当有自己的父类加载器。这里类加载器之间的父子关系一般不会以继承的关系来实现,而是都使用组合关系来复用父加载器的代码。
双亲委派模型的工作过程是:如果一个类加载器收到了类加载的请求,它首先不会自己去尝试加载这个类,而是把这个请求委派给父类加载器去完成,每一个层次的类加载器都是如此。
双亲委派模型对于保证 Java 程序的稳定运作很重要,它让 Java 类随着它的类加载器一起具备了一种带有优先级的层次关系。
双亲委派模型不是一个强制性的约束模型,而是 Java 设计者们推荐给开发者们的一种类加载器的实现方式。