文档章节

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

chunquedong
 chunquedong
发布于 2016/05/18 21:59
字数 875
阅读 1070
收藏 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

粉丝 45
博文 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,然后更新你的应用,就可以实现啦~可以在官网注册试用哦~
Android插件化开发,初入殿堂

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

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

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

技术小能手
07/20
0
0
Android插件化快速入门与实例解析(VirtualApk)必须了解一下

 集成一个第三方相册功能,只需集成一个插件APK到项目中,无需集成额外代码,并且支持随时更新相册功能,无需发布版本更新,无需AndroidManifest中声明四大组件,这就是插件化。   插件化...

android自学
07/22
0
0
Android插件化开发之动态加载技术学习

Android插件化开发之动态加载技术学习 为什么要插件化开发和动态加载呢?我认为原因有三点: 可以实现解耦 可以解除单个dex函数不能超过65535的限制 可以给apk瘦身,比如说360安全卫士,整个...

幸运券发放
05/18
0
0
我的Android重构之旅:插件化篇

随着项目的不断成长,即便项目采用了 MVP 或是 MVVM 这类优秀的架构,也很难跟得上迭代的脚步,当 APP 端功能越来越庞大、繁琐,人员不断加入后,牵一发而动全局的事情时常发生,后续人员如同...

codeGoogle
07/09
0
0

没有更多内容

加载失败,请刷新页面

加载更多

什么是集群、分布式、集中式、伪分布式

1. 集中式 将项目等部署到同一台机器上,对机器性能要求比较高,一般会用多台机器备份,否则,如果机器出现死机等状况,整个项目将不能运行。 eg:就好比你要盖一座房子,你房子就给一个人盖...

吴伟祥
35分钟前
0
0
微服务写的最全的一篇文章

今年有人提出了2018年微服务将疯狂至死,可见微服务的争论从未停止过。在这我将自己对微服务的理解整理了一下,希望对大家有所帮助。 1.什么是微服务 1)一组小的服务(大小没有特别的标准,...

Java烂猪皮
36分钟前
0
0
Java 并发编程原理

为什么要使用多线程? 通过多线程提高吞吐量 伸缩性比较好,可以增加 CPU 核心来提高程序性能 什么场景下使用多线程? 如:tomcat BIO Java 如何实现多线程? Thread、Runnable、ExecutorSer...

aelchao
51分钟前
2
0
谨慎的覆盖clone方法

说在前面 有些专家级程序员干脆从来不去覆盖clone方法,也从来不去调用它,除非拷贝数组。 其他方式 可以提供一个构造函数或者工厂去实现clone功能。 相比于clone,它们有如下优势: 不依赖于...

XuePeng77
52分钟前
1
0
什么是最适合云数据库的架构设计?

分布式数据库技术发展多年,但是在应用、业务的驱动下,分布式数据库的架构一直在不断发展和演进。 开源金融级分布式数据库SequoiaDB,经过6年的研发,坚持从零开始打造数据库核心引擎。在技...

巨杉数据库
今天
5
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部