文档章节

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

chunquedong
 chunquedong
发布于 2016/05/18 21:59
字数 875
阅读 1080
收藏 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插件化开发框架CJFrameForAndroid。 背景交代 首先,你需要知道什么是插件化开发。就拿最常见的QQ来说,在第三个界面动态那里有...

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

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

技术小能手
07/20
0
0
反插件化:你的应用不是一个插件(转)

本片文章主要翻译国外的一篇文章,原文在这里 摘要 是一种新型的的虚拟化框架,它允许移动应用程序在不安装应用程序的情况下动态加载并启动其应用程序。该技术的初衷是为了解决热修复问题和减...

隔壁老李头
10/03
0
0
Android插件化开发之动态加载技术学习

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

幸运券发放
05/18
0
0

没有更多内容

加载失败,请刷新页面

加载更多

防止Tweak

什么是tweak? 英文意思为捏, 拧,扭,稍稍调整(机器、系统等)。 依据维基百科的定义,tweak指的是对电子系统进行轻微调整来增强其功能的工具;在ios中tweak特指那些能够增强其它可执行程...

HeroHY
15分钟前
0
0
linux中常用标识---不定期更新

LINUX常用标识符: 1 & && | || &: 表示进程在后台运行 例如 redis-server & 不是所有后台运行都是& 比如es ./bin/elasticsearch -d es后台运行&&: 第一个命令执行成功后 才执行后面的命令...

geek土拨鼠
53分钟前
1
0
Mybatis 中$与#的区别,预防SQL注入

一直没注意Mybatis 中$与#的区别,当然也是更习惯使用#,没想到避免了SQL注入,但是由于要处理项目中安全渗透的问题,不可避免的又遇到了这个问题,特此记录一下。 首先是共同点: 在mybatis...

大雁南飞了
今天
0
0
Spring Clould负载均衡重要组件:Ribbon中重要类的用法

Ribbon是Spring Cloud Netflix全家桶中负责负载均衡的组件,它是一组类库的集合。通过Ribbon,程序员能在不涉及到具体实现细节的基础上“透明”地用到负载均衡,而不必在项目里过多地编写实现...

Ala6
今天
0
0
让 linux 删除能够进入回收站

可以参考这个贴子 https://blog.csdn.net/F8qG7f9YD02Pe/article/details/79543316 从那个git地址 把saferm.sh下载下来 把saferm.sh复制到 /usr/bin 目录下 在用~/目下 的.bashrc 下加一句这...

shzwork
今天
1
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部