Android---App 的安装过程

news2025/1/11 0:33:08

Android 系统中两个比较重要的服务 ActivityManagerService(AMS) 和 WindowManagerService(WMS),这篇文章中通过分析 apk 的安装过程,来了解 Android 中另一个比较重要的系统服务 -- PackageManagerService(PMS)

编译阶段

在分析安装过程之前,需要先了解 Android 项目是如何经过编译打包生成最终的 .apk 格式的安装包。Google 有一张官方图片来描述 apk 的打包流程。

 一个完整的项目可能包含多个 module,而每一个 module 中的内容可以分为2部分:

\bullet Resources 资源文件;

\bullet Java 或者 Kotlin 源代码。

因此整个项目的编译打包过程,也是针对这两部分来完成的。

编译 Resource 资源文件

资源文件包括项目中 res 目录下的各种 XML 文件、动画、drawable 图片、音视频等。AAPT 工具负责编译项目中的这些资源文件,所有资源文件会被编译处理。XML 文件(drawable 图片除外)会被编译成二进制文件。所以解压 apk 之后,无法直接打开 XML 文件。但是,assets 和 raw 目录下的资源并不会被编译,会被原封不动的打包到 apk 压缩包中

资源文件编译之后的产物包括两部分:

\bullet resource.arsc 文件(保存的是一个资源索引表);

\bullet R.java 文件(定义了各个资源 ID 常量)。

这两者结合就可以在代码中找到对应的资源引用。比如如下的 R.java 文件

可以看出 R.java 中的资源 id 是一个4字节的无符号整数,用16进制表示。其中最高的1字节(7f)表示 packageId,次高的一个字节(01)表示 typeId,最低的两个字节(0000)表示 entryId。

resources.arsc 相当于一个资源索引表,也可以理解为一个 map 映射表。其中,map 的 key 是 R.java 中的资源 ID,而 value 是其对应的资源所在路径。实际上 resources 描述上还有其它相关信息。关于 Resource.arsc 的解析可以参考:解析编译之后的 Resource.arsc 文件格式

编译源码部分

项目中的代码首先会通过 javac 编译为 .class 字节码文件,然后这些 .class 文件连同依赖的第三方库中的 .class 文件一同被 dx 工具优化为 .dex 文件。如果有分包,那么也可能会生成多个 .dex 文件。实际上,源代码文件也包括 AIDL 接口文件编译之后生成的 .java 文件。Android 项目中如果包含 .aidl 接口文件,这些 .aidl 文件会被编译成 .java 文件。

打包阶段

使用工具 APK Builder 将已经编译之后的 resource 和 .dex 文件一起打包到 apk 中。实际上被打包到 apk 中的还有一些其他资源。比如 AndroidManifest.xml 清单文件和第三方库中使用的动态库 .so 文件。

apk 创建好之后还不能直接使用,需要使用工具 jarsigner 对其进行签名。Android 系统不会安装没有签名的程序。签名之后会生成 META_INF 文件夹,此文件夹中保存着跟签名相关的各个文件:

\bullet CERT.SF:生成每个文件相对的密钥;

\bullet MANIFEST.MF:数字签名信息;

\bullet xxx.SF:JAR 文件的签名文件;

\bullet xxx.DSA:对输出文件的签名和公钥。

PMS 在安装过程中,会检测 apk 中的签名、证书的合法性。

按照常量,签名之后的 apk 应该是能正常安装使用了。但是,实际打包过程还会多一步 apk 优化操作。使用工具 zipalign 对 apk 中的未压缩资源(图片、视频)等进行对齐操作。让资源按照 4 字节的边界进行对齐,这种思想主要是为了加快资源的访问速度。如果每个资源的开始位置都是上一个资源之后的 4*n 字节,那么访问下一个资源就不用再遍历,直接跳到 4*n 字节处判断是不是一个新的资源即可。

至此,一个完整 apk 安装包就创建成功。一个完整 apk 解压缩之后内容,如下图所示

整个编译打包流程可以用下图来描述

PMS 将 apk 安装到手机设置

PMS 安装过程概览

当点击某一个 App 安装包进行安装时,首先会弹出一个系统界面指示我们进行安装操作,这个界面是 Android Framework 中预置的一个 Activity--PackageInstallerActivity.java。当点击安装后,PackageInstallerActivity 最终会将所安装的 apk 信息,通过 PackageInstallerSession 传给 PMS。

具体方法在 commitLocked() 方法中,如下代码所示

图中的 mPm 就是系统服务 PackageManagerService。installStage 方法就是正式开始 apk 的安装过程。整个 apk 的安装过程可以分为两大步:1)拷贝安装包;2)装载代码

拷贝安装包

从 installStage 方法开始看起,代码如下

图中1处创建了类型为 INIT_COPY的 message;图中2处创建了 InstallParams,并传入安装包的相关数据。

message 发送出去之后,由 PMS 的内部类 PackageHandler 接收并处理,如下代码所示

图中1处从 message 中取出 HandlerParams 对象;图中2处调用 connectToService() 方法连接安装 apk 的 service 服务。

PackageHandler 的 connectToService 方法

通过隐私 Intent 方法绑定 Service。实际绑定的 Service 类型是 DefaultContainerService 类型

当绑定 Service 成功之后,会在 onServiceConnected 方法中发送一个绑定操作的 message,如下所示

MCS_BOUND message 的接收者还是 PackageHandler,具体如下

mPendingInstall 是一个等待队列,里面保存所有需要安装 apk 解析出来的 handlerParams 参数。从 mPendingInstalls 中取出第一个需要安装的 HandlerParams 对象,并调用其 startCopy() 方法。在 startCopy() 方法中,继续调用一个抽象方法 handlerStartCopy() 处理安装请求。

通过之前的分析,我们知道 HandlerParams 实际类型是 InstallParams。因此,最终调用的是 InstallParams 的 handlerStartCopy 方法。

InstallParams 的 handlerStartCopy 方法

这个方法是整个安装包拷贝的核心方法。具体如下

图中1处设置安装标志位,决定是安装在手机内部存储空间还是 SD卡当中。图中2处判断 apk 安装位置,如果安装位置合法,则执行图中3处逻辑,创建一个 InstallArgs。实际上是其子类 FileInstallArgs 类型。然后调用其 copyApk() 方法进行安装包的拷贝工作。

FileInstallArgs 的 copyApk 方法

从图中可以看出,copyApk() 方法调用了 doCopyApk() 方法。doCopyApk 方法中主要做了3件事情。

图中1处创建存储安装包的目标路径,实际上是 dateApp应用包名目录;图中2处调用服务的 copyPackage 方法,将安装包 apk 拷贝到目标路径下;图中3处将 apk 中的动态库 so 文件也拷贝到目标路径中。

上图中的 IMediaContainerService 实际上就是在开始阶段进行连接操作的 DefaultContainerService 对象。其内部 copyPackage() 方法本质上就是执行 io 流操作,具体如下

最终,安装包在 dataApp 目录下以 base.apk 的方法保存。

至此,安装包的拷贝工作就已经完成。

装载代码

代码拷贝结束之后,就开始进入真正的安装步骤。代码回到上述的 HandleParams 的 startCopy 方法。

可以看出当安装包拷贝结束之后,继续调用 handleReturnCode() 方法来处理返回结果。最终调用 processPendingInstall() 方法处理安装过程,代码如下

图中1处执行预安装的操作,主要是检查安装包的状态,确保安装环境正常。如果安装环境有问题,则会清理拷贝文件;图中2处,是真正的安装阶段。installPackageTraceLI 方法中添加跟踪 trace,然后调用 installPackageLI 方法进行安装。图中3处处理安装完成之后的操作。installPackageLI 是 apk 安装阶段的核心代码

图中1处,调用 PackageParser 的 parsePackage 方法解析 apk 文件。主要是解析 AndroidManifest.xml 清单文件。将结果记录在 PackageParser.package 中。我们在清单文件中声明了 Activity、Service 等组件,就是在这一步被记录到 framework 中。后续它可以通过 startActivity 或 startService 启动相应的活动或者服务;图中2处对 apk 中的签名信息进行验证操作。collectCertificates 做签名验证,collectManifestDigest 主要做包的项目清单摘要的收集。主要是用来两个包是否一样,如果我们的设备上已经安装了一个 debug 版本的 apk,再次使用一个 release 版本的 apk 进行安装覆盖,会在这一步验证失败,会最终导致安装失败。图中3处执行 Dex 优化。图中4处调用 installNewPackageLI 方法,执行新 apk 的安装操作。

installNewPackageLI 方法,负责完成最后的 apk 安装过程。具体代码如下

scanPackageLI 继续扫描解析 apk 安装包文件,保存 apk 相关信息到 PMS 中,并创建 apk 的 data 目录,具体路径为datadata 应用包名。deletePackageLI 如果安装失败,则将安装包已经各种缓存文件删除。

至此,整个 apk 的安装过程结束。

安装成功之后,还会发送一个 App 安装成功的广播 ACTION_PACKAGE_ADDED。手机桌面应用注册了这个广播,当接收到应用安装成功之后,就将 apk 的启动 icon 显示在桌面上。

总结

主要介绍了一个 Android 项目从编译成 apk 文件,然后被安装到手机设备上的简要过程。编译分为:资源 + 源代码

生成 apk 之后还要经过签名、对齐等操作

apk 安装分2块进行:安装包拷贝代码装载

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1173635.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

项目实战:service业务逻辑组件引入

1、第一层DAO层 1.1、FruitDao接口 package com.csdn.fruit.dao; import com.csdn.fruit.pojo.Fruit; import java.util.List; //dao :Data Access Object 数据访问对象 //接口设计 public interface FruitDao {void addFruit(Fruit fruit);void delFruit(String fn…

Java--类和对象

目录 面向对象一.类1.类的创建默认初始化2.类的实例化3.注意事项利用类的创建来交换值 二.this1.使用this2.可使用this来调用其他构造方法来简化 三.构造方法3.1概念3.2特性3.3不带参数的构造方法3.4带参数的构造方法当使用自定义的构造方法后,再删除时,…

【Linux系统编程】系统用户和权限的操作

目录 一,Linux的用户 1,用户之间的切换 2,超级用户权限的使用 二,Linux的文件权限 1,文件信息的介绍 2,文件权限的修改 3,用户的修改 3-1,拥有者的更改 3-2,所属…

chinese-stable-diffusion中文场景文生图prompt测评集合

腾讯混元大模型文生图操作指南.dochttps://mp.weixin.qq.com/s/u0AGtpwm_LmgnDY7OQhKGg腾讯混元大模型再进化,文生图能力重磅上线,这里是一手实测腾讯混元的文生图在人像真实感、场景真实感上有比较明显的优势,同时,在中国风景、动…

“一键批量拆分HTML文本,高效整理文件,提升工作效率“

您是否曾经被大量的HTML文本文件困扰,难以找到所需的特定信息?现在,我们向您推荐一款强大的工具,它能够一键拆分HTML文本,让您轻松实现文件整理,提高工作效率! 首先,在首助编辑高手…

大航海时代Ⅳ 威力加强版套装 HD Version (WinMac)中文免安装版

《大航海时代》系列的人气SRPG《大航海时代IV》以HD的新面貌再次登场!本作品以16世纪的欧洲“大航海时代”为舞台,玩家将以探险家、商人、军人等不同身份与全世界形形色色的人们一起上演出跌宕起伏的海洋冒险。游戏中玩家的目的是在不同的海域中掌握霸权…

使用javafx,结合讯飞ai,搞了个ai聊天系统

第一步:先在讯飞ai那边获取接入的api 点进去,然后出现这个页面: 没有的话,就点击免费试用,有了的话,就点击服务管理: 用v2.0的和用3的都行,不过我推荐用2.0版本 文档位置&#xff1…

1825_ChibiOS的OSLIB中的存储分配器

全部学习汇总: GreyZhang/g_ChibiOS: I found a new RTOS called ChibiOS and it seems interesting! (github.com) 1. 之前有点不是很理解什么是静态OS,从这里基本上得到了这个答案。所谓的静态,其实就是内核之中不会使用Free以及Malloc这样…

ORAM文献笔记

Deterministic, stash-free write-only oram Roche D S, Aviv A, Choi S G, et al. Deterministic, stash-free write-only oram[C]//Proceedings of the 2017 ACM SIGSAC Conference on Computer and Communications Security. ACM, 2017: 507-521. 确定性的、无堆栈的只写O…

【黑马程序员】SSM框架——SpringBoot

文章目录 前言一、SpringBoot 简介1. 入门案例1.1 入门程序① 创建新模块② 选择当前模块需要使用的技术集③ 开发控制类④ 运行自动生成的 Application 类 1.2 创建 SpringBoot 程序的两种方式1.2.1 最简 SpringBoot 程序所包含的基础文件1.2.2 基于 SpringBoot 官网创建项目 …

解析SD-WAN组网方式及应用场景,全面了解典型案例

随着企业业务高速发展,跨区域开展业务首要解决的难题是构建各站点能互联互通的网络,然而目前大多数企业在广域网优化的问题上依旧碰壁,主要原因是企业广域网面临的挑战并不能马上得到解决。 传统网络互联方案无论是IPsec还是专线&#xff0c…

单目结构光三维重建最终公式推导

详细推导(建议自己推导一遍) 参考:https://blog.csdn.net/u010430651/article/details/104868734?spm1001.2014.3001.5502

【漏洞复现】Apache_Shiro_1.2.4_反序列化漏洞(CVE-2016-4437)

感谢互联网提供分享知识与智慧,在法治的社会里,请遵守有关法律法规 文章目录 1.1、漏洞描述1.2、漏洞等级1.3、影响版本1.4、漏洞复现1、基础环境2、漏洞分析3、漏洞验证 说明内容漏洞编号CVE-2016-4437漏洞名称Apache_Shiro_1.2.4_反序列化漏洞漏洞评级…

【Spring】bean的配置

文章目录 1. 前言2. name3. lazy-init4. init-method5. destroy-method6. factory-method和factory-bean 1. 前言 在之前的文章中.写到过bean的常用配置,当时只是介绍了bean标签中的常用属性配置:class,id和scope这三个属性. 不熟的小伙伴可以看一下这篇文章:【Spring】IOC容器…

类和对象 下

构造函数 初始化列表 构造函数是在构造类的时候给对象赋初值,并不是给类的成员函数初始化,赋值可以赋多次,而初始化只能初始化一次,这里我们引入初始化列表来对成员函数进行初始化 初始化列表,以冒号 :开…

【JVM系列】- 挖掘·JVM堆内存结构

挖掘JVM堆内存结构 文章目录 挖掘JVM堆内存结构堆的核心概念堆的特点 堆的内存结构内存划分新生代/新生区(Young Generation)老年代(Tenured Generation)永久代(或元数据区)(PermGen 或 MetaSpa…

学习视频剪辑:巧妙运用中画、底画,制作画中画,提升视频效果

随着数字媒体的普及,视频剪辑已经成为一项重要的技能。在视频剪辑过程中,制作画中画可以显著提升视频效果、信息传达和吸引力。本文讲解云炫AI智剪如何巧妙运用中画、底画批量制作画中画来提升视频剪辑水平,提高剪辑效率。 操作1、先执行云…

两种MySQL OCP认证应该如何选?

很多同学都找姚远老师说要参加MySQL OCP认证培训,但绝大部分同学并不知道MySQL OCP认证有两种,以MySQL 8.0为例。 一种是管理方向,叫:Oracle Certified Professional, MySQL 8.0 Database Administrator(我考试的比较…

“我“何去何从?2024年软件测试职业发展方向?运筹帷幄方能决胜千里之外...

目录:导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结(尾部小惊喜) 前言 测试的职责无非就…

《进化优化》第6章 进化策略

文章目录 6.1 (11) 进化策略6.2 1/5规则:推导走廊问题 6.3 μ1进化策略6.4 μλ和(μ,λ)进化策略6.5 自身自适应进化策略协方差阵自适应CMA 6.1 (11) 进化策略 假设f(x)是随机向量x的函数,我们想要最大化适应度f(x).…