LinuxGod.net
Linux大神网——精选每一篇高品质的技术干货
  1. 首页
  2. 开源快讯
  3. 正文

Android插件化框架DL代理Activity的原理和三个关键问题

2022年12月24日 87点热度

本文摘选自任玉刚著《Android开发艺术探索》,介绍了Android插件化技术的原理和三个关键问题,并给出了作者自己发起的开源插件化框架。

动态加载技术(也叫插件化技术)在技术驱动型的公司中扮演着相当重要的角色,当项目越来越庞大的时候,需要通过插件化来减轻应用的内存和CPU占用,还可以实现热插拔,即在不发布新版本的情况下更新某些模块。动态加载是一项很复杂的技术,这里主要介绍动态加载技术中的三个基础性问题,至于完整的动态加载技术的实现请参考笔者发起的开源插件化框架DL()。项目期间有多位开发人员一起贡献代码。

不同的插件化方案各有各的特色,但是它们都必须要解决三个基础性问题:资源访问、Activity生命周期的管理和ClassLoader的管理。 在介绍它们之前,首先要明白宿主和插件的概念,宿主是指普通的apk,而插件一般是指经过处理的dex或者apk,在主流的插件化框架中多采用经过特殊处 理的apk来作为插件,处理方式往往和编译以及打包环节有关,另外很多插件化框架都需要用到代理Activity的概念,插件Activity的启动大多 数是借助一个代理Activity来实现的。

1.资源访问

我们知道,宿主程序调起未安装的插件apk,一个很大的问题就是资源如何访问,具体来说就是插件中凡是以R开头的资源都不能访问了。这是因为宿主程序中并没有插件的资源,所以通过R来加载插件的资源是行不通的,程序会抛出异常:无法找到某某id所对应的资源。

针对这个问题,有人提出了将插件中的资源在宿主程序中也预置一份,这虽然能解决问题,但是这样就会产生一些弊端。首先,这样就需要宿主和插件同时持 有一份相同的资源linux常用命令,增加了宿主apk的大小;其次,在这种模式下,每次发布一个插件都需要将资源复制到宿主程序中android apk 动态加载linux驱动,这意味着每发布一个插件都要更新一下宿 主程序,这就和插件化的思想相违背了。

因为插件化的目的就是要减小宿主程序apk包的大小,同时降低宿主程序的更新频率并做到自由装载模块,所以这种方法不可取,它限制了插件的线上更新 这一重要特性。还有人提供了另一种方式,首先将插件中的资源解压出来,然后通过文件流去读取资源,这样做理论上是可行的,但是实际操作起来还是有很大难度 的。首先不同资源有不同的文件流格式,比如图片、XML等,其次针对不同设备加载的资源可能是不一样的,如何选择合适的资源也是一个需要解决的问题linux 下载,基于这两点,这种方法也不建议使用,因为它实现起来有较大难度。为了方便地对插件进行资源管理,下面给出一种合理的方式。

我们知道,Activity的工作主要是通过ContextImpl来完成的, Activity中有一个叫mBase的成员变量,它的类型就是ContextImpl。注意到Context中有如下两个抽象方法,看起来是和资源有关 的,实际上Context就是通过它们来获取资源的。这两个抽象方法的真正实现在ContextImpl中,也就是说,只要实现这两个方法android apk 动态加载linux驱动,就可以解决资源问题了。

14.2.png

下面给出具体的实现方式,首先要加载apk中的资源,如下所示

14.3.png

从loadResources()的实现可以看出,加载资源的方法是通过反射,通过调用AssetManager中的addAssetPath方 法,我们可以将一个apk中的资源加载到Resources对象中,由于addAssetPath是隐藏API我们无法直接调用,所以只能通过反射。下面 是它的声明,通过注释我们可以看出,传递的路径可以是zip文件也可以是一个资源目录,而apk就是一个zip,所以直接将apk的路径传给它,资源就加 载到AssetManager中了。然后再通过AssetManager来创建一个新的Resources对象,通过这个对象我们就可以访问插件apk中 的资源了,这样一来问题就解决了。

14.5.png

接着在代理Activity中实现getAssets()和getResources(),如下所示。关于代理Activity的含义请参看DL开源插件化框架的实现细节,这里不再详细描述了。

android 动态加载数据_android apk 动态加载linux驱动_android 动态加载apk

14.6.png

通过上述这两个步骤,就可以通过R来访问插件中的资源了。

2.Activity生命周期的管理

管理Activity生命周期的方式各种各样,这里只介绍两种:反射方式和接口方式。反射的方式很好理解,首先通过Java的反射去获取 Activity的各种生命周期方法,比如onCreate、onStart、onResume等,然后在代理Activity中去调用插件 Activity对应的生命周期方法即可,如下所示。

14.8.png

使用反射来管理插件Activity的生命周期是有缺点的,一方面是反射代码写起来比较复杂,另一方面是过多使用反射会有一定的性能开销。下面介绍 接口方式,接口方式很好地解决了反射方式的不足之处,这种方式将Activity的生命周期方法提取出来作为一个接口(比如叫DLPlugin),然后通 过代理Activity去调用插件Activity的生命周期方法,这样就完成了插件Activity的生命周期管理,并且没有采用反射,这就解决了性能 问题。同时接口的声明也比较简单,下面是DLPlugin的声明:

15.png

在代理Activity中只需要按如下方式即可调用插件Activity的生命周期方法,这就完成了插件Activity的生命周期的管理。

15.2.png

通过上述代码应该不难理解接口方式对插件Activity生命周期的管理思想,其中mRemoteActivity就是DLPlugin的实现。

3.插件ClassLoader的管理

为了更好地对多插件进行支持,需要合理地去管理各个插件的DexClassLoader,这样同一个插件就可以采用同一个ClassLoader去 加载类,从而避免了多个ClassLoader加载同一个类时所引发的类型转换错误。在下面的代码中,通过将不同插件的ClassLoader存储在一个 HashMap中,这样就可以保证不同插件中的类彼此互不干扰。

15.6.png

事实上插件化的技术细节非常多,这绝非一个章节的内容所能描述清楚的,另外插件化作为一种核心技术,需要开发者有较深的开发功底才能够很好地理解, 因此本节的内容更多是让读者对插件化开发有一个感性的了解,细节上还需要读者自己去钻研,也可以通过DL插件化框架去深入地学习。

本作品采用 知识共享署名 4.0 国际许可协议 进行许可
标签: activity android开发 android框架 apk 代理模式
最后更新:2022年12月24日

Linux大神网

每日更新,欢迎收藏♥ 不积跬步无以至千里,加油,共勉。

点赞
< 上一篇
下一篇 >

文章评论

您需要 登录 之后才可以评论

Linux大神网

每日更新,欢迎收藏♥
不积跬步无以至千里,加油,共勉。

最新 热点 随机
最新 热点 随机
英伟达将LinuxGPU内核模块作为开放源码/MIT双重许可证(图) virtualboxlinux官方版免费、小巧,同时支持Windows、Linux和Mac系统主机 Linux平台下的嵌入式软件要具备说明技能吗? 图解linuxlinux内核结构框图对内核框图有个总体的把握 linux虚拟机的安装步骤是什么?虚拟机和服务器的区别 精通Linux程序设计的高级程序员并非一件可望不可及的事情 linux虚拟机软件 Linux在安装的时候要注意什么?Linux安装注意事项 Python都有哪些应用,学习完成Python后能的工作方向 微软考虑用Rust开发内核的Linux内核-Hartman Linux内核和文件系统权限管理的基本用法——ACL权限 Linux内核启动有次引导程序Grub 小白在阿里云云服务器上如何发布自己的网站(建站|详细) 阿里云>社区>主题地图S>查看存储推荐 Linux的内核放在了哪里?/boot的启动目录一览 Linux中修改文件权限的命令、创建者所在组、所有人 linux到底难不难学呢?推荐可以查看Linux命令大全 Linux下修改文件权限的权限与所有权的实现就显得很有必要 国内性价比很高的Linux虚拟主机系统安装的流程是什么? 1.Linux磁盘分区和目录Linux发行版本之间的差别很少? 基于命令修改文件的权限命令-ld
Linux5.12的推送请求不断涌入新开放的合并窗口预计4月底看到它的稳定版本go语言被称作互联网时代的c语言,用来开发嵌入式linux的理由腾讯云服务器上也搭建一套环境,安装成功自动启动个人笔记本安装Ubuntu20.04LTS下载地址启动第一步--加载BIOS当你打开计算机电源(组图)虚拟机安装Ubuntu操作系统-Ubuntu空间20G镜像下载指令中各个make-C~/linuxM=`pwd编译Linux启动过程中的几个部分内核的引导(图)如何在Linux上安装虚拟机的结果大多都是怎么安装的一个免费软件时间跟进的小白鼠是什么鬼?专题计算机是如何启动的?、内核操作系统的启动流程学习Linux最简单、最实用的环境就是虚拟机环境(上)linux系统位数最简单的命令--linux基于命令修改文件的权限命令-lddf显示指定磁盘文件和目录的磁盘使用空间命令参数《构建高性能web站点》推荐理由及方法都能找到1.Linux磁盘分区和目录Linux发行版本之间的差别很少?centos系统版本详细信息解决方案与操作系统版本兼容的重要性分析国内性价比很高的Linux虚拟主机系统安装的流程是什么?Linux下修改文件权限的权限与所有权的实现就显得很有必要
Windows中命令提示符批量修改文件名称没有直接的方法是什么? 【1024】后端技术学习路线思维导图,看这一篇就够了! virtualbox安装virtualbox增强功能,虚拟机下载 虚拟主机技术极大怎么样?如何选择多个网站.Q. UNIX网络编程卷1:套接字联网API第3版pdf百度网盘下载地址 lsblk的大小扩容(2)-sh Linux内核虚拟内存管理的难点分析与解决办法(一) Linux计算机的防火墙必需运行OpenSSH服务器软件 Linux存储管理操作实践学号实验内容(9页珍藏版) 《深入linux设备驱动程序内核机制》(1)_社会万象_光明网(图) 【干货】Linux下的线程是什么?(二) 中科红旗Linux桌面操作系统V11上线,后续微软应该会做出调整 娃白天睡觉不太踏实,搞一个局域网实时监控,怎么办? 树莓派2正式发布支持微软Windows10操作系统派 关键词树莓派摄像系统自动跟踪语音控制和定位模块结合用户识别 嵌入式Linux软件设计系统设计的一般流程是什么?(图) 图解linuxlinux内核结构框图对内核框图有个总体的把握 Linux的内核放在了哪里?/boot的启动目录一览 利用FinalShell访问虚拟机FinalShell 如何查看linux查找文件夹命令(图)查看文件内容
标签聚合
命令模式 shell sudo unix linux系统 虚拟机 linux服务器 电脑 文件目录 软件
书籍
课程
技术群
技术干货大合集↓
  • 2023年2月 / 1篇
  • 2023年1月 / 161篇
  • 2022年12月 / 187篇
  • 2022年11月 / 76篇

COPYRIGHT © 2023 linuxgod.net ALL RIGHTS RESERVED.