Android插件化动态加载原理(二)
Android插件化动态加载原理(二)
chunquedong 发表于2年前
Android插件化动态加载原理(二)
  • 发表于 2年前
  • 阅读 1004
  • 收藏 24
  • 点赞 2
  • 评论 3

腾讯云 技术升级10大核心产品年终让利>>>   

摘要: 本篇介绍Android插件开发中的类加载器

继续前文,介绍Android插件开发中的类加载器。

类加载器介绍

类加载是指把Java类从字节码文件读到JVM中。Java通过类加载器机制可以在运行时动态加载Java字节码。开发者可以通过自定义ClassLoader类,来实现加载并运行网络上下载的字节码文件。

Android对Java的ClassLoader类进行了扩展,不止加载jar文件,还可以使用DexClassLoader来加载apk文件中的字节码。DexClassLoader的构造函数的第一个参数就是apk文件路径:

DexClassLoader(String dexPath, String dexOutputDir, String libPath, ClassLoader parent)

我们通过自定义的ClassLoader来加载运行插件,这也是所有插件开发库之所以能实现的基础。

JVM(Android中不叫JVM,以下只是统称)中可以有多个类加载器,JVM系统启动时有默认的类加载器。所有的类加载器都有一个父类加载器,类加载器通过这种父亲后代的方式组织在一起,形成树状层次结构。 一般情况下类加载器在加载某个类之前,会首先代理给其父类加载器。当父类加载器找不到的时候,才会尝试自己去加载。最后如果都没有加载成功则抛出ClassNotFoundException异常。 这个所谓的“双亲委派”的意思就是父类加载器优先加载。

类加载器问题

在JVM中只有类名称和加载它的类加载器,两者完全一样时,才被认为两个类的是相同的。即便是同样的字节码,如果被两个不同的类加载器加载后,得到的Java类也是不同的。 如果一个类由不同就类加载器加载,则转换的时候会出现ClassCastException。按照双亲委派的原则,这样就能保证公用的代码使用同一个类加载器加载。

双亲委派原则是推荐的做法,自定义类加载器的时候也可以违反这个原则。这样可以很容易构建一个隔离空间,不同版本的代码能同时运行,而相互不影响。这个特性有时候很有用。

如果插件和宿主有相同的类,是应该优先加载宿主中的类还是优先加载宿主中的类呢:

  1. 插件优先时,可能会导致宿主和插件的类被隔离到不同的空间,在相互调用时可能会产生ClassCastException异常。
  2. 宿主优先时,在Android4.4以下系统,如果插件和宿主有相同的类,在某些情况下可能会出现Class ref in pre-verified异常。详见这里

所以最好不要在插件和宿主中打包相同的类。

替换Android的默认类加载器

为了能简化插件系统的使用,我们不显式的使用自定义类加载器,而是使用反射的方法偷偷替换系统原有的ClassLoader,而且把系统原有的ClassLoader作为自定义插件类加载器的父亲。 示例代码如下:

ClassLoader old = (ClassLoader) Reflection.getField(loadedApk, "mClassLoader");
myClassLoader = new MyAppClassLoader(old, this);
Reflection.setField(loadedApk, "mClassLoader", myClassLoader);

替换为自己的DexClassLoader就可以加载apk文件中的代码了。剩下的内容下篇再写。

请关注axbasePlugin插件化开发框架Github

共有 人打赏支持
chunquedong
粉丝 47
博文 115
码字总数 34233
作品 4
评论 (3)
pennymei
OneAPM 详悉 Android Apps 性能信息,协助你快速定位性能瓶颈,快速捕捉性能表现差的应用代码,发现错误并随时向你发送警报,这一切只需下载并安装 OneAPM Android SDK,然后更新你的应用,就可以实现啦~可以在官网注册试用哦~
维维诺
您好,Suppressed: java.lang.ClassNotFoundException: Didn't find class "com.testplugin.MainActivity" on path: DexPathList[[zip file "/data/app/info.axbase.apphost-1/base.apk"],nativeLibraryDirectories=[/data/app/info.axbase.apphost-1/lib/arm64, /vendor/lib64, /system/lib64]] 出现这个错误是什么问题啊?
chunquedong

引用来自“维维诺”的评论

您好,Suppressed: java.lang.ClassNotFoundException: Didn't find class "com.testplugin.MainActivity" on path: DexPathList[[zip file "/data/app/info.axbase.apphost-1/base.apk"],nativeLibraryDirectories=[/data/app/info.axbase.apphost-1/lib/arm64, /vendor/lib64, /system/lib64]] 出现这个错误是什么问题啊?
这个错误是在什么情况下发生的,方便的话把代码发给我看看chunquedong (at) 163.com
×
chunquedong
如果觉得我的文章对您有用,请随意打赏。您的支持将鼓励我继续创作!
* 金额(元)
¥1 ¥5 ¥10 ¥20 其他金额
打赏人
留言
* 支付类型
微信扫码支付
打赏金额:
已支付成功
打赏金额: