热门标签 | HotTags
当前位置:  开发笔记 > 前端 > 正文

浅谈AndroidClassloader动态加载分析

这篇文章主要介绍了浅谈AndroidClassloader动态加载分析,详细的介绍了ClassLoader概念、分类,具有一定的参考价值,有兴趣的可以了解一下

ClassLoader概念

我们知道,Java源文件(.java)经过编译器编译之后,会转换成Java字节码(.class),然而程序是如何加载这些字节码文件到内存中呢?这就用到了ClassLoader,即类加载器。ClassLoader类加载器负责读取 Java 字节代码,并转换成 java.lang.Class类的一个实例。从而只有class文件被载入到了内存之后,才能被其程序所引用。所以ClassLoader就是用来动态加载class文件到内存当中用的。

ClassLoader的分类

Android中的常用几种类加载器类型继承关系划分可以用一组关系图来表示

BootClassLoder

通过查看ClassLoader源码 我们得知,Android中在默认父加载器传入的情况下,默认父加载器为PathClassLoder,而PathClassLoader的父加载器正是BootClassLoader。BootClassLoader是ClassLoader的内部类,是包内可见,我们无法直接使用,也无法直接动态加载。

 /**
   * Encapsulates the set of parallel capable loader types.
   */
  private static ClassLoader createSystemClassLoader() {
    String classPath = System.getProperty("java.class.path", ".");
    String librarySearchPath = System.getProperty("java.library.path", "");

    ...省略部分代码

    //默认父构造器为PathClassLoder
    return new PathClassLoader(classPath, librarySearchPath, BootClassLoader.getInstance());
  }

URLClassLoader

URLClassLoader继承自SecureClassLoader,SecureClassLoader继承自ClassLoader。URLClassLoader的特点就是只能加载jar文件,但是dalvik不能直接识别jar。所以在Android中无法直接使用这个类加载器。

BaseDexClassLoader

BaseDexClassLoader直接继承自ClassLoader,下面是其构造函数

public BaseDexClassLoader(String dexPath, File optimizedDirectory, String librarySearchPath, ClassLoader parent) {
    super(parent);
    this.pathList = new DexPathList(this, dexPath, librarySearchPath, null);
    ....
  }

下面解析下这四个参数

  1. dexPath:指目标类所在的apk、dex或者jar文件的路径(包括SD卡),然后加载器将从该路径中寻找到指定的目标类。当然了,这个路径可以是多个路径,这样可以寻找到多个目标类,多路径之间需要使用特定的分隔符,分隔符可以使用System.getProperty("path.separtor")获取。
  2. optimizedDirectory:由于dex文件被包含在apk或者jar文件中,需要先解压出来,而这个参数 就代表了被解压的路径。而且apk文件其实也是一个压缩包,解压的过程其实也是一个ODEX优化的过程,那么何为ODEX优化呢?其实就是把包里面的可执行程序提取出来变成ODEX文件,存放到optimizedDirectory目录下,因为提取出来的原因,应用第一次进行启动的时候,直接使用ODEX文件 启动速度自然是比解压再启动速度是要快的。为什么是应用第一次启动呢?因为dex版本只有第一次启动会解压执行程序到/data/dalvik-cache(针对PathClassLoader),或者optimizedDirectory文件目录下(针对DexClassLoader),之后就可以直接读取目录下的dex文件了。
  3. librarySearchPath:指的是目标类所使用的c、c++库存放的路径
  4. parent:是指该加载器的父加载器,一般为当前执行类的加载器。

PathClassLoader

public PathClassLoader(String dexPath, ClassLoader parent) { 
    super(dexPath, null, null, parent); 
} 

 public PathClassLoader(String dexPath, String librarySearchPath, ClassLoader parent) {
    super(dexPath, null, librarySearchPath, parent);
   ...
 }

通过源码我们可以知道,PathClassLoader继承于BaseDexClassLoader,并且构造器将optimizedDirectory置为null,也就是没有设置ODEX优化后的存储路径,前文有提到,如果没有设置optimizedDirectory目录,那么默认存储路径就是/data/dalvik-cache。因为这个原因,PathClassLoader被设定成只能加载Android系统类和已安装的android应用类。

DexClassLoader

public DexClassLoader(String dexPath, String optimizedDirectory, String librarySearchPath, ClassLoader parent) {
    super(dexPath, new File(optimizedDirectory), librarySearchPath, parent);
   ...
  }

DexClassLoader也是继承于BaseDexClassLoader,支持加载包含classes.dex文件的apk、jar,前文我们提到dalvik不支持直接加载jar文件,那么为什么到了DexClassLoader这里怎么就可以支持加载了呢?原因在于其父类BaseDexClassLoader对于“.jar”,“.apk”,".zip",".dex"后缀的文件都会进行对应的处理,最终提取成可执行的dex文件。然而URLClassLoader并未对此做类似的处理,因此我们一般会采用DexClassLoader做动态加载。

InMemoryDexClassLoader

 public InMemoryDexClassLoader(ByteBuffer[] dexBuffers, ClassLoader parent) {
    super(dexBuffers, parent);
  }
  
  public InMemoryDexClassLoader(ByteBuffer dexBuffer, ClassLoader parent) {
    this(new ByteBuffer[] { dexBuffer }, parent);
  }

InMemoryDexClassLoader继承于BaseDexClassLoader,是API26新增的类加载器。dexBuffers数组构造了一个DexPathList,可用于加载内存中的dex。

DelegateLastClassLoader

public DelegateLastClassLoader(String dexPath, ClassLoader parent) {
    super(dexPath, parent);
  }

DelegateLastClassLoader是API27新增的类加载器,继承自 PathClassLoader。DelegateLastClassLoader实行最后的查找策略。使用DelegateLastClassLoader来加载每个类和资源,使用的是以下顺序:

  1. 判断是否已经加载过该类
  2. 搜索此类的类加载器是否已经加载过该类
  3. 搜索与此类加载器相关联的dexPath文件列表,并委托给父加载器。

双亲委托机制

Android类加载器通过loadClass加载目标类,下面是加载的源码

 protected Class<&#63;> loadClass(String name, boolean resolve)
    throws ClassNotFoundException
  {
      // 首先检查当前目标类是否已经被加载过,有则直接返回
      Class<&#63;> c = findLoadedClass(name);
      if (c == null) {
        try {
          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) {
          //如果仍未找到,则通过寻找子ClassLoader的目标类(如果子ClassLoader重写了findClass)
          c = findClass(name);
        }
      }
      return c;
  }

由上述源码,我们可以总结:

  1. 当前类加载器首先检查目标类是否已经被加载过,有则直接返回
  2. 当前类加载器会先委托父类加载器加载目标类,如果未设置父加载器,则检查辅助加载器是否支持查询加载目标类
  3. 只有上述加载器找不到目标类的时候,才会调用当前类加载器(Child) 查询路径寻找目标类。

以上这么做的好处是:一方面防止目标类的重复加载,另外一方面 主要考虑安全因素,防止有人重写原生类,比如说java.lang.String这样的数据类型,替换原生的String类,加载到JVM中,造成严重的安全问题。

双亲委托机制 在Android热修复领域中也有着广泛的应用。每个ClassLoader可以有多个dex文件,每个dex文件是一个Element,多个dex文件组成一个dexElements,类加载器寻找类的时候,会遍历dexElements中的dex文件,再通过dex文件遍历目标类。由于双亲委托机制的存在,寻找到目标类后就直接返回,不再寻找其他dex文件下该目标类,热修复的原理就是hook住ClassLoader,使其先加载修复后的目标类,而存在的BUG的目标类不会被加载。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • Android中高级面试必知必会,积累总结
    本文介绍了Android中高级面试的必知必会内容,并总结了相关经验。文章指出,如今的Android市场对开发人员的要求更高,需要更专业的人才。同时,文章还给出了针对Android岗位的职责和要求,并提供了简历突出的建议。 ... [详细]
  • 视图分区_组复制常规操作网络分区amp;混合使用IPV6与IPV4 | 全方位认识 MySQL 8.0 Group Replication...
    网络分区对于常规事务而言,每当组内有事务数据需要被复制时,组内的成员需要达成共识(要么都提交,要么都回滚)。对于组成员资格的变更也和保持组 ... [详细]
  • 本文内容皆为作者原创,如需转载,请注明出处:https:www.cnblogs.comxuexianqip13045462.html1.自定义分页器的拷贝及使用当我们需要使用 ... [详细]
  • 入门Java需熟练掌握哪些技术呢?
    Java工程师无疑是当下令人艳羡的工作之一,因此,每年都有大批大批的朋友想要报名Java工程师学习Java也就不奇怪了。那么入门Java掌握哪些技术能力 ... [详细]
  • 这是原文链接:sendingformdata许多情况下,我们使用表单发送数据到服务器。服务器处理数据并返回响应给用户。这看起来很简单,但是 ... [详细]
  • SpringBoot uri统一权限管理的实现方法及步骤详解
    本文详细介绍了SpringBoot中实现uri统一权限管理的方法,包括表结构定义、自动统计URI并自动删除脏数据、程序启动加载等步骤。通过该方法可以提高系统的安全性,实现对系统任意接口的权限拦截验证。 ... [详细]
  • Java序列化对象传给PHP的方法及原理解析
    本文介绍了Java序列化对象传给PHP的方法及原理,包括Java对象传递的方式、序列化的方式、PHP中的序列化用法介绍、Java是否能反序列化PHP的数据、Java序列化的原理以及解决Java序列化中的问题。同时还解释了序列化的概念和作用,以及代码执行序列化所需要的权限。最后指出,序列化会将对象实例的所有字段都进行序列化,使得数据能够被表示为实例的序列化数据,但只有能够解释该格式的代码才能够确定数据的内容。 ... [详细]
  • Centos7.6安装Gitlab教程及注意事项
    本文介绍了在Centos7.6系统下安装Gitlab的详细教程,并提供了一些注意事项。教程包括查看系统版本、安装必要的软件包、配置防火墙等步骤。同时,还强调了使用阿里云服务器时的特殊配置需求,以及建议至少4GB的可用RAM来运行GitLab。 ... [详细]
  • 本文介绍了新款奇骏的两个让人上瘾的功能,分别是智能互联系统和BOSE音响。通过对新款奇骏的配置和功能进行评测,探讨了这两个新增功能的使用体验和优势。此外,还介绍了新款奇骏的其他配置和改进,如增加的座椅和驾驶辅助系统,以及内饰的舒适性提升。对于喜欢音响的消费者来说,BOSE音响的升级也是一个亮点。最后,文章提到了BOSE音响的数字还原能力,以及7座版无法配备BOSE音响的原因。 ... [详细]
  • 本文介绍了adg架构设置在企业数据治理中的应用。随着信息技术的发展,企业IT系统的快速发展使得数据成为企业业务增长的新动力,但同时也带来了数据冗余、数据难发现、效率低下、资源消耗等问题。本文讨论了企业面临的几类尖锐问题,并提出了解决方案,包括确保库表结构与系统测试版本一致、避免数据冗余、快速定位问题等。此外,本文还探讨了adg架构在大版本升级、上云服务和微服务治理方面的应用。通过本文的介绍,读者可以了解到adg架构设置的重要性及其在企业数据治理中的应用。 ... [详细]
  • 禁止程序接收鼠标事件的工具_VNC Viewer for Mac(远程桌面工具)免费版
    VNCViewerforMac是一款运行在Mac平台上的远程桌面工具,vncviewermac版可以帮助您使用Mac的键盘和鼠标来控制远程计算机,操作简 ... [详细]
  • 本文详细介绍了云服务器API接口的概念和作用,以及如何使用API接口管理云上资源和开发应用程序。通过创建实例API、调整实例配置API、关闭实例API和退还实例API等功能,可以实现云服务器的创建、配置修改和销毁等操作。对于想要学习云服务器API接口的人来说,本文提供了详细的入门指南和使用方法。如果想进一步了解相关知识或阅读更多相关文章,请关注编程笔记行业资讯频道。 ... [详细]
  • 这篇“Webpack是怎么工作的”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大 ... [详细]
  • 我正在使用sql-serverkafka-connect和debezium监视sqlserver数据库,但是当我发布并运行我的wo ... [详细]
  • ConsumerConfiguration在kafka0.9使用JavaConsumer替代了老版本的scalaConsumer。新版的配置如下:bootstrap. ... [详细]
author-avatar
多米音乐_34176403
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有