当前位置:   article > 正文

Android类加载器

android类加载器

Java中类加载器 

1.启动类加载器(Bootstrap ClassLoader):这个类加载器负责放在<JAVA_HOME>\lib目录中的,或者被-Xbootclasspath参数所指定的路径中的,并且是虚拟机识别的类库,用户无法直接使用。

2.扩展类加载器(Extension ClassLoader):这个类加载器由sun.misc.Launcher$AppClassLoader实现。它负责<JAVA_HOME>\lib\ext目录中的,或者被java.ext.dirs系统变量所指定的路径中的所有类库。用户可以直接使用。

3.应用程序类加载器(Application ClassLoader):这个类由sun.misc.Launcher$AppClassLoader实现。是ClassLoader中getSystemClassLoader()方法的返回值。它负责用户路径(ClassPath)所指定的类库。用户可以直接使用。如果用户没有自己定义类加载器,默认使用这个。

4.自定义加载器:用户自己定义的类加载器。

双亲委托模式

某个特定的类加载器在接到加载类的请求时,首先将加载任务委托给父类加载器,每一层的加载器都采用这种方式,直到委托给顶层的启动类加载器为止,如果超类无法加载该类,则会将类的加载内容退回给它的下一层。

  • 双亲委托模式:可以避免重复加载,能有效的确保一个类的全局唯一性
  • 如果不使用这种委托模式,那我们就可以随时使用自定义的类来动态替代一些核心的类,存在非常大的安全隐患,比如定义 java.lang.String 替代系统的String等操作。

Android中类加载器

Android中包含以下几种ClassLoader

  • BootClassLoader:用来加载Framework层的字节码文件
  • URLClassLoader:加载.jar文件和文件夹中的class,javaWeb等使用,谷歌不用
  • BaseDexClassLoader:PathClassaLoader、DexClassLoader父类
  • PathClassaLoader:加载已经安装到系统中的apk中的class文件
  • DexClassLoader:加载指定目录中的字节码文件(包括aar,apk,jar)

BaseDexClassLoader为核心类,androidStudio中是看不到BaseDexClassLoader源码的,提供下源码查看地址:BaseDexClassLoader.java

BaseDexClassLoader加载过程

1.构造参数

DexClassLoader是BaseDexClassLoader子类,下面分析参数。

  1. /**
  2. * 构造方法
  3. */
  4. public class BaseDexClassLoader extends ClassLoader {
  5. public BaseDexClassLoader(String dexPath, File optimizedDirectory,
  6. String librarySearchPath, ClassLoader parent) {
  7. }
  8. }
  9. /**
  10. * 构造方法
  11. */
  12. public class DexClassLoader extends BaseDexClassLoader {
  13. public DexClassLoader(String dexPath, String optimizedDirectory,
  14. String librarySearchPath, ClassLoader parent) {
  15. super(dexPath, null, librarySearchPath, parent);
  16. }
  17. }

dexPath:apk/dex/jar文件路径

optimizedDirectory:是odex将dexPath路径中dex优化后的输出路径,这个路径必须是手机内部路劲。此参数已弃用,自API级别26起不再生效。

librarySearchPath:需要加载的C/C++库路径

parent:父加载器(这个比较重要与Android加载class的机制有关)

2.加载过程

BaseDexClassLoader构造方法中传入的参数最终会传给DexPathList,BaseDexClassLoader.findClass最终调用DexPathList.findClass。

  1. /**
  2. * 构造方法
  3. */
  4. public BaseDexClassLoader(String dexPath, File optimizedDirectory,
  5. String librarySearchPath, ClassLoader parent, boolean isTrusted) {
  6. super(parent);
  7. //DexPathList
  8. this.pathList = new DexPathList(this, dexPath, librarySearchPath, null, isTrusted);
  9. if (reporter != null) {
  10. reportClassLoaderChain();
  11. }
  12. }
  13. /**
  14. * BaseDexClassLoader.findClass
  15. */
  16. @Override
  17. protected Class<?> findClass(String name) throws ClassNotFoundException {
  18. List<Throwable> suppressedExceptions = new ArrayList<Throwable>();
  19. //DexPathList.findClass
  20. Class c = pathList.findClass(name, suppressedExceptions);
  21. if (c == null) {
  22. ClassNotFoundException cnfe = new ClassNotFoundException(
  23. "Didn't find class \"" + name + "\" on path: " + pathList);
  24. for (Throwable t : suppressedExceptions) {
  25. cnfe.addSuppressed(t);
  26. }
  27. throw cnfe;
  28. }
  29. return c;
  30. }
  • DexPathList.findClass最终调用element.findClass,Element[] dexElements是由makeDexElements方法进行赋值。
  • Thinker热修复方案就是将补丁dex插入到dexElements最前端,这样classLoader就会先加载补丁中修复了bug的class文件,由于classLoader双亲委托,再加载原先有bug的class文件时,发现已经有一摸一样的修复了bug的class被加载了,就会直接返回不会再去加载旧class文件,从而完成修复bug的目的。
  1. /**
  2. * List of dex/resource (class path) elements.
  3. * Should be called pathElements, but the Facebook app uses reflection
  4. * to modify 'dexElements' (http://b/7726934).
  5. */
  6. private Element[] dexElements;
  7. /**
  8. * DexPathList.findClass
  9. */
  10. public Class<?> findClass(String name, List<Throwable> suppressed) {
  11. //dexElements数组
  12. for (Element element : dexElements) {
  13. Class<?> clazz = element.findClass(name, definingContext, suppressed);
  14. if (clazz != null) {
  15. return clazz;
  16. }
  17. }
  18. if (dexElementsSuppressedExceptions != null) {
  19. suppressed.addAll(Arrays.asList(dexElementsSuppressedExceptions));
  20. }
  21. return null;
  22. }
  23. /**
  24. * dexElements初始化赋值
  25. */
  26. private static Element[] makeDexElements(List<File> files, File optimizedDirectory,
  27. List<IOException> suppressedExceptions, ClassLoader loader, boolean isTrusted) {
  28. Element[] elements = new Element[files.size()];
  29. int elementsPos = 0;
  30. /*
  31. * Open all files and load the (direct or contained) dex files up front.
  32. */
  33. for (File file : files) {
  34. if (file.isDirectory()) {
  35. // We support directories for looking up resources. Looking up resources in
  36. // directories is useful for running libcore tests.
  37. elements[elementsPos++] = new Element(file);
  38. } else if (file.isFile()) {
  39. String name = file.getName();
  40. DexFile dex = null;
  41. if (name.endsWith(DEX_SUFFIX)) {
  42. // Raw dex file (not inside a zip/jar).
  43. try {
  44. dex = loadDexFile(file, optimizedDirectory, loader, elements);
  45. if (dex != null) {
  46. elements[elementsPos++] = new Element(dex, null);
  47. }
  48. } catch (IOException suppressed) {
  49. System.logE("Unable to load dex file: " + file, suppressed);
  50. suppressedExceptions.add(suppressed);
  51. }
  52. } else {
  53. try {
  54. dex = loadDexFile(file, optimizedDirectory, loader, elements);
  55. } catch (IOException suppressed) {
  56. /*
  57. * IOException might get thrown "legitimately" by the DexFile constructor if
  58. * the zip file turns out to be resource-only (that is, no classes.dex file
  59. * in it).
  60. * Let dex == null and hang on to the exception to add to the tea-leaves for
  61. * when findClass returns null.
  62. */
  63. suppressedExceptions.add(suppressed);
  64. }
  65. if (dex == null) {
  66. elements[elementsPos++] = new Element(file);
  67. } else {
  68. elements[elementsPos++] = new Element(dex, file);
  69. }
  70. }
  71. if (dex != null && isTrusted) {
  72. dex.setTrusted();
  73. }
  74. } else {
  75. System.logW("ClassLoader referenced unknown path: " + file);
  76. }
  77. }
  78. if (elementsPos != elements.length) {
  79. elements = Arrays.copyOf(elements, elementsPos);
  80. }
  81. return elements;
  82. }
声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/不正经/article/detail/247728
推荐阅读
相关标签
  

闽ICP备14008679号