JVM类加载机制分析

1. 类加载机制

在之前的文章万字长文把[JVM]从头到尾讲一遍
中详细讲了JVM虚拟机的内存结构和垃圾回收算法,如下图概述了JVM整个的运行机制,今天来讲讲最上面的类加载机制部分

JVM
  • 什么是类加载机制?
    Java虚拟机将编译后的.class文件加载到内存中,进行校验、转换、解析和初始化,到最终的使用,这就是类的加载机制。类的加载时机并未有明确的规定,但是类明确了类的初始化时机。

2. 类加载机制的过程

类的加载机制大致分为五个过程:加载、验证、准备、解析、初始化。下面将分析每个过程中做了什么事。

2.1 加载

通过ClassLoader加载一个Class对象到内存中。具体过程:

  1. 通过全限定名获取此类的二进制字节流(.class文件),至于二进制字节流在哪里获取并没有限制,可以从jar、apk、zip、数据库、网络、自己运行生成都可以。
  2. 在内存中生成一个代表此类的java.lang.Class对象,并作为方法区这个类的访问入口。这个Class对象并没有规定放在Java堆中,有些虚拟机将它放在方法区中。

2.2 验证

验证加载后的类是否符合.Class文件结构,类数据是否符合虚拟机的要求,确保不会危害虚拟机的安全。具体过程如下:

  1. 文件格式验证:验证二机制字节流是否符合.class的文件格式,并且验证该.class文件是否在虚拟机的处理范围内,文件格式验证合格后才将二进制的数据存放在内存的方法区中。
  2. 元数据验证:主要是对该类的元数据信息进行语义检查,保证不存在不符合 Java 语义规范的元数据信息。
  3. 字节码验证:主要对类的方法体进行验证,确保类的方法不会做出危害虚拟机的行为
  4. 符号引用验证:对类本身饮用其他类型的验证,包括对全限定名是否能找到对应的类,是否能找到对应的类的方法和字段,访问性是否合适。

2.3 准备

  1. 对于类变量(static修饰)为其分配内存,并赋值初始值(如0,false)。
  2. 对于常量(final修饰)为其赋值设置的数值。

2.4 解析

将类符号引用转换成直接引用。

2.5 初始化

给类变量(static)赋值,并执行static{}方法。这里的触发执行的方法是类构造器<clinit>中。

  1. 类构造器<clinit>是编译器自己生成的,它会按类的顺序的收集类变量和静态代码块,如果一个类中没有类变量也没有静态代码块将没有类构造器。它和实例构造器<init>是不同。
  2. 父类的构造器将优先于子类的构造器执行。子接口的构造器不需要调用父类的类构造器。
  3. 静态代码块可以访问出现在它前面的静态变量,但不能访问后面的静态变量,只可以赋值。

2.5.1类初始化的时机

  1. new一个对象的时候;获取和设置static的变量和方法的时候;
  2. 使用 java.lang.reflect 包对方法进行反射调用的时候。
  3. 当一个类的父类没被初始化时,会优先初始化父类。
  4. 当虚拟机启动时,需要指定一个要执行的主类时候。
  5. 当使用 JDK 1.7 的动态语言支持时,如果一个 java.lang.invoke.MethodHandle 实例最后的解析结果 REF_getStatic、REF_putStatic、REF_invodeStatic 的方法句柄,并且这个方法句柄所对应的类没有进行过初始化,则需要先触发其初始化。

3. 类加载器ClassLoader

这里讲解的ClassLoader是安卓的类加载器,不是Java的加载器,这是有区分的,比如Java的类加载器加载的是Jar里面的.class文件的集合,而安卓则是将.class文件的集合全部写入到一个dex文件中,删除一些重复的代码,以此来提高性能。

3.1 ClassLoader的类型

Android的类加载器类型也可以分为两种:

  1. 系统类加载器
  2. 自定义类加载器

无论哪种加载器,它们都要继承ClassLoader这个抽象父类。其中系统类加载器主要有:BootClassLoader、PathClassLoader、DexClassLoader

3.1.1 BootClassLoader

class BootClassLoader extends ClassLoader {

    private static BootClassLoader instance;

    @FindBugsSuppressWarnings("DP_CREATE_CLASSLOADER_INSIDE_DO_PRIVILEGED")
    public static synchronized BootClassLoader getInstance() {
        if (instance == null) {
            instance = new BootClassLoader();
        }

        return instance;
    }
...

}

BootClassLoader继承于ClassLoader,它是一个没有父加载器的加载器,它在Zygote进程启动的时候,BootClassLoader加载器将会被创建,用它加载一些预加载类,方便以后fork进程时复用资源。同时它也是ClassLoader的内部类。

3.1.2 PathClassLoader

public class PathClassLoader extends BaseDexClassLoader {

    /**
     * @param dexPath : Dex相关文件的路径
     * @param parent  : 父加载器
     */
    public PathClassLoader(String dexPath, ClassLoader parent) {
        super(dexPath, null, null, parent);
    }

    /**
     * @param dexPath: Dex相关文件的路径
     * @param librarySearchPath:包含C/C++库的路径集合
     * @param parent : 父加载器
     */
    public PathClassLoader(String dexPath, String librarySearchPath, ClassLoader parent) {
        super(dexPath, null, librarySearchPath, parent);
    }
    ...
}

PathClassLoader继承BseDexClassLoader,同时BaseDexClassLoader继承ClassLoader。
PathClassLoader的创建在system_server进程中,PathClassLoader类加载器通常加载已经安装的apk的dex文件。
PathClassLoader类加载器默认的解压的dex文件的存储路径是:/data/dalvik_cache路径中。
如下是创建的时机:

public class ZygoteInit {

    // 创建完system_server进程后,会执行此方法
    private static Runnable handleSystemServerProcess(ZygoteArguments parsedArgs) {
        if (systemServerClasspath != null) {
            //...
        } else {
            ClassLoader cl = null;
            // 创建PathClassLoader加载器
            if (systemServerClasspath != null) {
                cl = createPathClassLoader(systemServerClasspath, parsedArgs.mTargetSdkVersion);
            }
        }
    }

    static ClassLoader createPathClassLoader(String classPath, int targetSdkVersion) {
        String libraryPath = System.getProperty("java.library.path");

        // 父加载器是BootClassLoader
        ClassLoader parent = ClassLoader.getSystemClassLoader().getParent();

        // 创建工厂模式创建PathClassLoader
        return ClassLoaderFactory.createClassLoader(classPath, libraryPath, libraryPath,
                parent, targetSdkVersion, true /* isNamespaceShared */, null /* classLoaderName */);
    }

}


public abstract class ClassLoader {

    public static ClassLoader getSystemClassLoader() {
        return SystemClassLoader.loader;
    }

    static private class SystemClassLoader {
        public static ClassLoader loader = ClassLoader.createSystemClassLoader();
    }

    private static ClassLoader createSystemClassLoader() {
        String classPath = System.getProperty("java.class.path", ".");
        String librarySearchPath = System.getProperty("java.library.path", "");
        // 父加载器是BootClassLoader
        return new PathClassLoader(classPath, librarySearchPath, BootClassLoader.getInstance());
    }

}

可见PathClassLoader的父加载器是BootClassLoader

3.1.3 DexClassLoader

public class DexClassLoader extends BaseDexClassLoader {

    /**
     *
     * @param dexPath : Dex相关文件的路径
     * @param optimizedDirectory: 解压的dex的存储路径
     * @param librarySearchPath:包含C/C++库的路径集合
     * @param parent : 父加载器
     */
    public DexClassLoader(String dexPath, String optimizedDirectory,
                          String librarySearchPath, ClassLoader parent) {
        super(dexPath, null, librarySearchPath, parent);
    }
}

DexClassLoader也是继承BaseDexClassLoader,相比PathClassLoader则是可以定义解压dex的存储路径。

除了BootClassLoader、PathClassLoader、DexClassLoader这三个类加载器外还有,InMemoryDexClassLoader:用于加载内存的dex;SecureClassLoader:权限检查的ClassLoader;URLClassLoader:URL的ClassLoade。

Android ClassLoader关系

3.2 ClassLoader的加载过程

Android中所有的类加载器都继承于ClassLoader抽象类,这个类的loadClass()方法同样实现了双亲委托机制。

3.2.1 双亲委托机制

public abstract class ClassLoader {

   /**
     * 双亲委托机制
     */
    protected Class<?> loadClass(String name, boolean resolve)
            throws ClassNotFoundException {
        // 1. 先检查class是否已经加载过
        Class<?> c = findLoadedClass(name);
        if (c == null) {
            // 没有加载过
            try {
                if (parent != null) {
                    // 先给父ClassLoader加载Class
                    c = parent.loadClass(name, false);
                } else {
                    // 调用BootClassLoader加载Class
                    c = findBootstrapClassOrNull(name);
                }
            } catch (ClassNotFoundException e) {
                // ClassNotFoundException thrown if class not found
                // from the non-null parent class loader
            }


            if (c == null) {
                // 父的ClassLoader都没有加载class,则调用findClass()给此ClassLoader加载
                c = findClass(name);
            }
        }
        return c;
    }


    protected Class<?> findClass(String name) throws ClassNotFoundException {
        throw new ClassNotFoundException(name);
    }

}

ClassLoader的loadClass()方法定义了加载器加载类的过程:

  1. 如果class文件已经加载过,则从缓存中找。
  2. 否则给递归给父加载器的loadClass()方法查找class文件
  3. 如果父加载器没找到,则调用自己的findClass(name)方法开始找class文件。
    这种设计避免了一些核心类的加载被用户自定义复写,导致功能不同。

那这个findClass()方法在ClassLoader中是一个空实现,它让给你子类去实现这个查找的过程。那这里以BaseDexClassLoader为例,看findClass()是如何查找class文件的:

public class BaseDexClassLoader extends ClassLoader {

    private final DexPathList pathList;

    public BaseDexClassLoader(String dexPath,
                              String librarySearchPath, ClassLoader parent, ClassLoader[] sharedLibraryLoaders,
                              boolean isTrusted) {
        super(parent);
        this.sharedLibraryLoaders = sharedLibraryLoaders == null
                ? null
                : Arrays.copyOf(sharedLibraryLoaders, sharedLibraryLoaders.length);
        this.pathList = new DexPathList(this, dexPath, librarySearchPath, null, isTrusted);

        reportClassLoaderChain();
    }

    @Override
    protected Class<?> findClass(String name) throws ClassNotFoundException {
        //  调用DexPathList.findClass方法查找class
        Class c = pathList.findClass(name, suppressedExceptions);
        if (c == null) {
            ClassNotFoundException cnfe = new ClassNotFoundException(
                    "Didn't find class \"" + name + "\" on path: " + pathList);
            for (Throwable t : suppressedExceptions) {
                cnfe.addSuppressed(t);
            }
            throw cnfe;
        }
        return c;
    }


}

调用DexPathList.findClass()方法去查找class文件

public final class DexPathList {
    private Element[] dexElements;

    DexPathList(ClassLoader definingContext, String dexPath,
                String librarySearchPath, File optimizedDirectory, boolean isTrusted) {
         this.dexElements = makeDexElements(splitDexPath(dexPath), optimizedDirectory,
                suppressedExceptions, definingContext, isTrusted);
    }

    public Class<?> findClass(String name, List<Throwable> suppressed) {
        // 遍历Element数组去查询
        for (Element element : dexElements) {
            Class<?> clazz = element.findClass(name, definingContext, suppressed);
            if (clazz != null) {
                return clazz;
            }
        }

        if (dexElementsSuppressedExceptions != null) {
            suppressed.addAll(Arrays.asList(dexElementsSuppressedExceptions));
        }
        return null;
    }

}


    /*package*/ static class Element {
        @UnsupportedAppUsage
        private final File path;
    
        private final Boolean pathIsDirectory;

        @UnsupportedAppUsage
        private final DexFile dexFile;

        private ClassPathURLStreamHandler urlHandler;
        private boolean initialized;

        @UnsupportedAppUsage
        public Element(DexFile dexFile, File dexZipPath) {
            if (dexFile == null && dexZipPath == null) {
                throw new NullPointerException("Either dexFile or path must be non-null");
            }
            this.dexFile = dexFile;
            this.path = dexZipPath;
            this.pathIsDirectory = (path == null) ? null : path.isDirectory();
        }

        public Class<?> findClass(String name, ClassLoader definingContext,
                                  List<Throwable> suppressed) {
             //  调用DexFile.loadClassBinaryName()方法去查找
            return dexFile != null ? dexFile.loadClassBinaryName(name, definingContext, suppressed)
                    : null;
        }
}

public final class DexFile {

    public Class loadClassBinaryName(String name, ClassLoader loader, List<Throwable> suppressed) {
        return defineClass(name, loader, mCookie, this, suppressed);
    }

    private static Class defineClass(String name, ClassLoader loader, Object cookie,
                                     DexFile dexFile, List<Throwable> suppressed) {

        Class result = null;
        try {
            result = defineClassNative(name, loader, cookie, dexFile);
        }
        ...
        return result;
    }

    private static native Class defineClassNative(String name, ClassLoader loader, Object cookie,
                                                  DexFile dexFile)
        
}

DexPathList有一个Element[]数组,每个Element有dex文件路径,通过遍历Element[]数组,调用Element. loadClassBinaryName()方法去查找是否对应的class文件,最后调用defineClassNative()native方法去查找。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,311评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,339评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,671评论 0 342
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,252评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,253评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,031评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,340评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,973评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,466评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,937评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,039评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,701评论 4 323
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,254评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,259评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,485评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,497评论 2 354
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,786评论 2 345

推荐阅读更多精彩内容