• Android类加载


    类加载机制
    • ClassLoader 抽象类

    • BootCLassLoader 用于加载Android Frameword层的class文件

    • BaseDexClassLoader 父类

    • PathClassLoader 子类 Android应用程序类加载器

    • DexClassLoader 子类 额外提供的动态类加载器

    PathClassLoader loadClass方法

    protected Class loadClass(String name, boolean resolve)
            throws ClassNotFoundException
        {
            synchronized (getClassLoadingLock(name)) {
                // First, check if the class has already been loaded
                /*第一先判断这个类有没有被加载过
                 如果加载过,返回该类的Class对象
                 没有加载过返回 null          
                */
                Class c = findLoadedClass(name);
                //如果这个类没有加载过
                if (c == null) {
                    long t0 = System.nanoTime();
                    try {
                    	//先看父类加载器可不可以加载,parent == null代表着使用根类加载器加载
                        if (parent != null) {
                            c = parent.loadClass(name, false);
                        } else {
                            c = findBootstrapClassOrNull(name);
                        }
                    } catch (ClassNotFoundException e) {
                        // ClassNotFoundException thrown if class not found
                        // from the non-null parent class loader
                    }
    				//如果父类加载器加载不了,再用本类加载器加载
                    if (c == null) {
                        // If still not found, then invoke findClass in order
                        // to find the class.
                        long t1 = System.nanoTime();
                        c = findClass(name);
    
                        // this is the defining class loader; record the stats
                        sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
                        sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
                        sun.misc.PerfCounter.getFindClasses().increment();
                    }
                }
                //resolve为true进行类加载链接操作,反之不进行
                if (resolve) {
                    resolveClass(c);
                }
                return c;
            }
        }
    
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    1. 先在自己中查找,自己没有,去父类BaseDexClassLoader中查找,依然没有
    2. 去ClassLoader抽象类中查找,找到loadClass方法
    3. loadClass第一步找缓存,缓存去native方法中查找。
    4. 缓存中没找到,通过parent(成员对象的父类加载器)调用loadClass的方法,PathClassLoader的parent是BootCLassLoader
    5. 如果从parent(BootClassLoader)中查找,没有找到。就使用自己的查找
    6. findClass方法找到自己的
    双亲委派机制

    某个类加载器在加载类时,首先将加载任务委托给父类加载器,一次递归,如果父类加载器可以完成类加载任务,就成功返回;只有父类加载器无法完成此加载任务或者没有父类加载器时,才自己去加载。

    为什么使用双亲委派机制?

    1. 避免重复加载,当父加载器已经加载了该类的时候,就没有必要子ClassLoader再加载一次。
    2. 安全性考虑,防止核心API库被随意篡改。
    热修复

    大致流程

    把出问题的类打包成dex文件

    因为classloader加载类是找到第一个就返回了,所以我们可以把dex文件插到dexElement数组前边,

    在application的attchbasecontext方法中进行插桩

    利用反射进行插桩

    1. 获取到当前应用的PathClassloader

    2. 反射获取到DexPatchList属性对象pathList

    3. 反射修改pathList的dexElements

    a. 把补丁包patch.dex转化为Element

    b. 获得pathList的dexElements属性(old)

    c. patch+old合并,并反射赋值给pathList的dexElements

  • 相关阅读:
    JDK、eclipse软件的安装
    [第九篇]——Docker 镜像使用
    2022卡塔尔世界杯小组赛出线形势分析指南——德国队会被西班牙做掉吗?
    2023-09-09青少年软件编程(C语言)等级考试试卷(一级)解析
    【k8s】(一) 使用kubeadm快速部署一个K8s集群
    爬某网站延禧宫率第一集视频
    精智屏如何实现秒级计划任务
    Word文档里有空白但无法打字怎么回事?
    操作系统:了解操作系统(编译、操作系统管理、进程、线程)
    2024双非网安捡漏华五0854经验分享
  • 原文地址:https://blog.csdn.net/qq_31309689/article/details/126464449