文档章节

Android插件化动态加载原理(二)

chunquedong
 chunquedong
发布于 2016/05/18 21:59
字数 875
阅读 1083
收藏 24

继续前文,介绍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

chunquedong

粉丝 46
博文 131
码字总数 36559
作品 4
海淀
高级程序员
私信 提问
加载中

评论(3)

chunquedong
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
维维诺
维维诺
您好,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]] 出现这个错误是什么问题啊?
pennymei
pennymei
OneAPM 详悉 Android Apps 性能信息,协助你快速定位性能瓶颈,快速捕捉性能表现差的应用代码,发现错误并随时向你发送警报,这一切只需下载并安装 OneAPM Android SDK,然后更新你的应用,就可以实现啦~可以在官网注册试用哦~
知识总结 插件化学习 Activity加载分析

现在安卓插件化已经很成熟,可以直接用别人开源的框架实现自己项目,但是学习插件化的实现原理是安卓研发工程师加深安卓系统理解的很好途径。 安卓插件化学习 插件Activity加载方式分析 实现...

常兴E站
2017/05/19
0
0
Android 插件化 动态升级

最新内容请见原文:Android 插件化 动态升级 不少朋友私信以及 Android开源交流几个 QQ 群 中都问到这个问题,这里简单介绍下 1、作用 大多数朋友开始接触这个问题是因为 App 爆棚了,方法数...

Trinea
2014/09/19
3.2K
1
加载assets下APK文件 及快速加载插件APK里面的资源

Android 系统为每个新设计的程序提供了/assets目录,这个目录保存的文件可以打包在程序里。/res 和/assets的不同点是,android不为/assets下的文件生成ID。如果使用/assets下的文件,需要指定...

desaco
01/09
0
0
Android插件化开发,初入殿堂

好久没有写博客了,这次准备写写我这几天的研究成果——Android插件化开发框架CJFrameForAndroid。 背景交代 首先,你需要知道什么是插件化开发。就拿最常见的QQ来说,在第三个界面动态那里有...

kymjs张涛
2014/10/12
0
9
有关Android插件化的一些总结思考

最近几年移动开发业界兴起了「 插件化技术 」的旋风,各个大厂都推出了自己的插件化框架,各种开源框架都评价自身功能优越性,令人目不暇接。随着公司业务快速发展,项目增多,开发资源却有限...

技术小能手
2018/07/20
0
0

没有更多内容

加载失败,请刷新页面

加载更多

多表查询

第1章 多表关系实战 1.1 实战1:省和市  方案1:多张表,一对多  方案2:一张表,自关联一对多 1.2 实战2:用户和角色 (比如演员和扮演人物)  多对多关系 1.3 实战3:角色和权限 (比如...

stars永恒
31分钟前
0
0
求推广,德邦快递坑人!!!!

完全没想好怎么来吐槽自己这次苦逼的德邦物流过程了,只好来记一个流水账。 从寄快递开始: 2019年1月15日从 德邦物流 微信小app上下单,截图如下: 可笑的是什么,我预约的是17号上门收件,...

o0无忧亦无怖
昨天
4
0