这里只是学习源码,基于android api27。关于对ClassLoader的理解可以学习一看你就懂,超详细java中的ClassLoader详解
ClassLoader是一个负责加载类的对象。通常情况下,ClassLoader根据提供的类名去文件系统查找相应的.class文件,生成类结构。
每个类对象都用一个定义它的ClassLoader的引用。
数组类型的类对象不是由ClassLoader创建的,而是根据需要由Java运行时自动创建的。数组类型通过getClassLoader
方法获取到的ClassLoader对象和它的元素的ClassLoader相同;如果元素是原始类型,那么数组类型的ClassLoader为null。如下:
public class JavaDemo {
public static void main(String[] args) {
System.out.println(JavaDemo[].class.getClassLoader());
System.out.println(int[].class.getClassLoader());
}
}
结果为:
sun.misc.Launcher$AppClassLoader@7d4991ad
null
从上面提到的博客中我们可以学习到ClassLoader的“双亲委托”模型。JVM内置了“bootstrap class loader”,它是最底层的ClassLoader,可以作为其他ClassLoader的parent(这里并不是父类,而是ClassLoader.getParent方法得到的值)。
支持并发加载类的ClassLoader被称为有并行能力(parallel capable)的ClassLoader。这类ClassLoader需要在初始化的时候通过registerAsParallelCapable
方法来注册自身,ClassLoader默认会调用该方法。其子类如果是有并行能力的,也需要调用该方法。在双亲委托不是很严格的情况下,ClassLoader需要具备并行能力,否则加载类可能会导致死锁,因为loadClass
方法在加载类的过程中会持有加载器的锁。
loadClass
该方法用于加载类。上面的博客可以深入学习。
getResource
public URL getResource(String name) {
URL url;
if (parent != null) {
url = parent.getResource(name);
} else {
url = getBootstrapResource(name);
}
if (url == null) {
url = findResource(name);
}
return url;
}
该方法用于获取资源。由源码可知,和loadClass方法原理大致相同,先交由parent查找,找不到在通过findResource方法查找。若需覆写,只需要覆写findResource
方法即可。
getResourceAsStream
public InputStream getResourceAsStream(String name) {
URL url = getResource(name);
try {
return url != null ? url.openStream() : null;
} catch (IOException e) {
return null;
}
}
与getResource
的不同在于返回一个InputStream对象。
getSystemClassLoader
public static ClassLoader getSystemClassLoader() {
return SystemClassLoader.loader;
}
返回系统的ClassLoader,这也是新建的ClassLoader实例的默认parent,同时也是启动应用的ClassLoader。
该方法最早在运行时启动时被调用,同时被设置为Thread的上下文ClassLoader(Thread.setContextClassLoader)。
对于ClassLoader的默认构造函数,就是将该函数的返回值设置为parent:
protected ClassLoader() {
this(checkCreateClassLoader(), getSystemClassLoader());
}
protected ClassLoader(ClassLoader parent) {
this(checkCreateClassLoader(), parent);
}
private ClassLoader(Void unused, ClassLoader parent) {
this.parent = parent;
}
而SystemClassLoader.loader
的值,由一下函数获取:
private static ClassLoader createSystemClassLoader() {
String classPath = System.getProperty("java.class.path", ".");
String librarySearchPath = System.getProperty("java.library.path", "");
return new PathClassLoader(classPath, librarySearchPath, BootClassLoader.getInstance());
}
由加载路径猜测应该是AppClassLoader。打印一下:
sun.misc.Launcher$AppClassLoader@7d4991ad
证实了我们的猜测。