赞
踩
任何一个 Java 程序都是由一个或多个 class 文件组成,在程序运行时,需要将 class 文件加载到 JVM 中才可以使用,负责加载这些 class 文件的就是 Java 的类加载机制。ClassLoader 的作用简单来说就是加载 class 文件,提供给程序运行时使用。每个 Class 对象的内部都有一个 classLoader 字段来标识自己是由哪个 ClassLoader 加载的。
Android 的类加载机制是基于 Java 的类加载机制,但在 Android 平台上有一些特殊的实现。
BootClassLoader:
这是 Android 系统中的根类加载器,负责加载 Android 系统核心库。它是一个原生代码实现的类加载器,其加载路径在操作系统级别上提前配置好。(用于加载Android Framework层class文件)
PathClassLoader:
这是应用程序类加载器,负责加载应用程序的代码和资源。每个应用程序都有自己独立的 PathClassLoader 实例,用于加载应用程序 APK 中的类。(可以加载指定的dex,以及jar、zip、apk中的classes.dex)
DexClassLoader:
这是一个特殊的类加载器,用于加载未安装在应用程序目录下的 dex 文件。DexClassLoader 可以指定 dex 文件的路径,加载其中的类。(用于加载指定的dex,以及jar、zip、apk中的classes.dex)
PathClassLoader 与 DexClassLoader 的共同父类是 BaseDexClassLoader 。
可以看到两者唯一的区别在于:创建 DexClassLoader 需要传递一个 optimizedDirectory 参数,并且会将其创建为 File 对象传给 super ,而 PathClassLoader 则直接给到null。因此两者都可以加载指定的dex,以及jar、zip、apk中的classes.dex
PathClassLoader pathClassLoader = new PathClassLoader("/sdcard/xx.dex",getClassLoader());
File dexOutputDir = context.getCodeCacheDir();
DexClassLoader dexClassLoader = new DexClassLoader("/sdcard/xx.dex",dexOutputDir.getAbsolutePath(), null,getClassLoader());
其实, optimizedDirectory 参数就是dexopt的产出目录(odex)。那 PathClassLoader 创建时,这个目录为null,就意味着不进行dexopt?并不是, optimizedDirectory 为null时的默认路径为:/data/dalvik-cache。
Java
双亲委托机制(Parent Delegation Model)是 Java 类加载机制中的一种实现方式。它是一种层次化的类加载器结构
,通过一种层级关系来管理和组织类的加载。
在双亲委托机制下,每个类加载器都有一个父类加载器
(除了顶层的启动类加载器)。当一个类加载器需要加载某个类时,它首先将加载请求委托给父类加载器。只有当父类加载器无法加载该类时,子类加载器才会尝试加载。
具体的执行过程如下:
1.当一个类加载器收到加载请求时,它首先检查是否已经加载了这个类。如果已经加载,则直接返回已加载的类。
2.如果该类未被加载过,则将加载请求委托给其父类加载器。
3.父类加载器继续按照同样的规则进行加载:首先检查是否已经加载了这个类,然后委托给它的父类加载器。
这个委托的过程一直持续到达到最顶层的启动类加载器。如果启动类加载器仍无法加载这个类,则子类加载器开始尝试加载。
子类加载器尝试加载这个类,如果成功加载则返回,否则抛出 ClassNotFoundException 异常。
这种层级的双亲委托机制可以确保类的一致性和安全性。它的核心思想是优先使用高层次的父类加载器去加载类
,这样可以避免重复加载和类的冲突。如果一个类已经被父类加载器加载了,那么子类加载器就不需要再加载,直接使用父类加载器加载的类即可。
Java 中的类加载器通常按照以下顺序进行委托:
启动类加载器(Bootstrap ClassLoader) ->
扩展类加载器(Extension ClassLoader) ->
应用程序类加载器(Application ClassLoader)。
开发者也可以自定义类加载器,并根据需要覆盖父类加载器的加载行为。
总的来说,双亲委托机制通过层级的类加载器结构,确保类的加载顺序和一致性,防止类的重复加载和冲突。它是 Java 类加载机制的一种重要实现方式,保证了 Java 程序的稳定性和安全性。
Android
在 Android 中,双亲委托机制采用了类似于 Java 的双亲委托模型,但在顶层类加载器的实现上有所不同。Android 的类加载器结构具体如下:(再来一遍)
BootClassLoader(引导类加载器):
这是 Android 系统中的根类加载器,负责加载核心库(例如核心 Java 类库和 Android 运行时库)。它是一个原生代码实现的类加载器,并且被固化在 Android 系统中,不允许应用程序开发者直接使用。
PathClassLoader(路径类加载器):
它是 Android 应用程序类加载器的最主要实现。每个应用程序都有自己独立的 PathClassLoader 实例,用于加载应用程序 APK 中的类和资源。该加载器会搜索应用程序的 APK 文件路径,并加载其中的类。
DexClassLoader(DEX 类加载器):
这是一种特殊的类加载器,可用于加载未安装在应用程序目录下的 DEX 文件。通过指定 DEX 文件的路径,可以加载其中的类。
Android 的类加载器在加载类时会按照以下顺序进行委托:
在 Android 应用程序中,PathClassLoader 是最常用的类加载器
。它负责加载应用程序的代码和资源,包括 APK 文件中的 Java 类、独立库(如 .so 文件)和其他资产。PathClassLoader 沿着类路径搜索并加载类,如果找不到类,则委托给父类加载器进行加载。
通过双亲委托机制,Android 的类加载器确保了类的唯一性和一致性
。如果一个类已经被父类加载器加载了,子类加载器就不会再次加载该类,直接使用已加载的类实例。这样可以避免重复加载和类的冲突
。
需要注意的是,为了适应 Android 运行环境,Android 的类加载器做了一些特殊处理。例如,在加载资源时,可以使用 Context 类提供的 getResources() 方法加载资源并返回一个由 PathClassLoader 加载的资源对象。
总结起来,Android 的双亲委托机制通过类加载器的层级关系,确保了类的加载顺序和一致性。PathClassLoader 是主要的应用程序类加载器,负责加载 APK 文件中的类和资源。BootClassLoader 是根类加载器,负责加载核心库。该机制保证了 Android 应用程序的稳定性和安全性。
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。