ClassLoader

作者: sollian | 来源:发表于2018-06-01 14:37 被阅读102次

这里只是学习源码,基于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

证实了我们的猜测。

相关文章

网友评论

    本文标题:ClassLoader

    本文链接:https://www.haomeiwen.com/subject/fmltsftx.html