Android---StartActivity启动过程

在手机桌面应用中点击某一个 icon 之后,最终是通过 startActivity 去打开某一个 Activity 页面。我们知道,Android 中的一个 APP 就相当于一个进程。所以,startActivity 操作中还需要判断,目标 Activity 的进程是否已经创建。如果没有,则在显示 Activity 之前还需要将进程 Process 提前创建出来。

假设是从 Activity A 跳转到另一个 APP 中的 Activity B。过程如下图所示

整个 startActivity 的流程分为 3 大部分,也涉及 3 个进程之间的交互:

\bullet ActivityA --> ActivityManagerService(简称 AMS)

\bullet ActivityManagerService --> ApplicationThrad

\bullet ApplicationThread --> Activity

1. ActivityA 到 AMS 阶段

这一过程并不复杂,用如下这张图表示具体过程

接下来查看源码,看看做了哪些操作

Activity 的 startActivity(),最终调用了 startActivityForResult() 方法。传入的 -1 表示不需要获取 startActivity 的结果。

Activity 的 startActivityForResult(),具体代码如下所示。调用 Instrumentation 的 execStartActivity方法,剩下的交给 Instrumentation 类去处理。

Instrumentation 类主要用来监控应用程序与系统交互。蓝线部分的 MainThread 是 ActivityThread 类型。ActivityThread 可用理解为1个进程,这里即表示 ActivityA 所在的进程。通过 MainThread 获取一个 ApplicationThread 引用,这个引用就是实现进程间通信的。具体来说,就是 AMS 所在的系统进程通知应用进程进行了一系列操作。

Instrumentation 类的 execStartActivity。在 instrumentation 中会通过 ActivityManager.getService获取 AMS 的实例。然后,调用其 startActivity 方法。实际上这里就是通过 AIDL 来调用 AMS 的 startActivity 方法。

至此,startActivity 的工作重心成功的从进程 A 转移到了系统进程 AMS中。

2. ActivityManagerService 到 ApplicationThread 阶段

下面来探索在 AMS 中是如何一步一步执行到 B 进程的。

上面在讲 Instrumentation 时,我们提到了一个 ApplicationThread 类来负责进程间通信的。这里 AMS 最终其实就是调用了 B 进程的一个 ApplicationThread 引用,从而间接地通知 B 进程进行相应操作。

相比较 startActivity 到 AMS,AMS 到 ApplicationThread 的流程看起来复杂了很多。实际上,这里面就干了两件事情:

a):综合处理 launchMode 和 Intent 中的 Flag 标志位,并根据处理结果生成一个目标 Activity B 的对象(ActivityRecord);

b):判断是否需要为目标 Activity B 创建一个新的进程(ProcessRecord)、新的任务栈(TaskRecord)

AMS 的 startActivity

从上图中的代码可以看出,经过多个方法的调用,最终通过 obtainStarter() 方法获取了 ActivityStarter 类型的对象,然后调用其 execute() 方法。在 execute() 方法中,会再次调用其内部的 startActivityMayWait 方法。

ActivityStarter 的 startActivityMayWait 

ActivityStarter 类专门负责一个 Activity 的启动操作主要作用包括解析 Intent、创建 ActivityRecord、如果有可能还要创建 TaskRecord。

startActivityMayWait 的部分实现代码如下

从上面代码可以看出,获取目标 Activity 信息的操作由 mSupervisor 来实现,它是 ActivityStackSupervisor 类型。主要是负责 Activity 所处栈的管理类。在上面代码的 resolveIntent 中,实际上是调用系统 PackageManagerService 来获取最佳 Activity。有时候,我们通过隐式 Intent 启动 Activity 时,系统中可能存在多个 Activity 可以处理 Intent。此时会弹出一个选择框让用户选择具体需要打开哪一个 Activity 界面,就是此处的逻辑处理结果。

在 startActivityMayWait 方法中,调用了一个重载的 startActivity 方法,最终会调用 ActivityStarter 中的 startActivityUnchecked 方法来获取启动 Activity 的结果。

ActivityStarter 的 startActivityUnchecked

图中 1 处计算启动 Activity 的 Flag 值;图中2处处理 Task 和 Activity 的进栈操作;图中3处启动栈中顶部的 Activity。

\bullet computeLaunchingTaskFlags 方法

这个方法的主要作用是计算启动 Activity 的 Flag,不同的 Flag 决定了启动 Activity 最终会被放置到哪一个 Task 集合中。

图中1处 mInTask 是 TaskRecord 类型,此处为 null,代表 Activity 要加入的栈不存在。因此需要判断是否需要新建 Task;图中2处的 mSourceRecord 是 ActivityRecord 类型,它是用来描述初始 Activity(比如,ActivityA 启动了 ActivityB,ActivityA 就是初始 Activity)。当我们使用 Context 或者 Application 启动 Activity 时,此时 mSourceRecord 为 null;图中3处表示初始 Activity,如果是在 LAUNCH_SINGLE_INSTANCE 栈中的 Activity,需要添加 FLAG_ACTIVITY_NEW_TASK 的标识。因为 LAUNCH_SINGLE_INSTANCE 栈只能允许保存一个 Activity;图中4处表示,如果 LaunchMode 设置了 LAUNCH_SINGLE_TASK 或 LAUNCH_SINGLE_INSTANCE,则也要创建一个新栈。

\bullet startActivityLocked 方法

方法中会调用 insertTaskAtTop() 方法,尝试将 Task 和 Activity 入栈。如果 Activity 是以 NEW_TASK 的模式启动或者 Task 堆栈中不存在该 TaskId,则 task 会重新入栈并且放在栈的顶部。注意:Task 先入栈,之后才是 Activity 入栈,它们是包含关系。

上面一下子引出了好几个概念:Stack,Task,Activity。其实它们都是在 AMS 内部维护的数据结构,它们之间的关系如下图

\bullet  resumeFocusedStackTopActivityLocked 方法

经过一些列调用,最终代码又回到了 ActivitSupervisor 中

ActivityStackSupervisor 的 startSpecificActivityLocked 方法

图中1处根据进程名称和 Application 的 UID,来判断目标进程是否已经创建。如果没有,则代表进程未创建;图中2处调用 AMS 创建 Activity 所在进程。

不管是目标进程已经存在还是新建目标进程,最终都会调用图中红线部分的 realStartActivityLocked 方法来执行启动 Activity 的操作。

ActivityStackSupervisor 的 realStartActivityLocked 方法

这个方法在 Android 27 和 28两个版本的区别很大。从 android-28 开始 Activity 的启动交给了事务(Transaction)来完成。图中1处创建了 Activity 启动事务,并传入 app.thread 参数,它是 applicationThread 类型。在上文 startActivity 阶段,applicationThread 是为例实现进程间通信的,是 ActivityThread 的一个内部类;图中2处执行 Activity 的启动事务,由 ClientLifecycleManager 来完成。具体代码如下

可以看出,实际上是调用了启动事务的 ClientTransaction 的 schedule() 方法。而这个 transaction 实际上是在创建 ClientTransaction 时,传入的 app.thread 对象,也就是 applicationThrad。

这里传入的 app.thread 会赋值给 ClientTransaction 的成员变量 mClient,ClientTransaction 会调用 mClient.scheduleTransaction(this) 来执行事务。这个 app.thread 是 ActivityThread 的内部类 ApplicationThread,所以事务最终是调用 app.thread 的 scheduleTransaction 执行

 到这里位置,startActivity 的操作就成功的从 AMS 转移到了另一个进程B 中的 ApplicationThread 中。剩下的就是 AMS 通过进程间通信机制通知 ApplicationThread 执行 ActivityB 的生命周期方法。

3. ApplicationThread 到 Activity 阶段

上面我们分析到,AMS 将启动 Activity 的任务作为一个事务 ClientTransaction 去完成。在 ClientLifecycleManager 中会调用 ClientTransaction 的 schedule() 方法。如下代码所示

mClient 是一个 IApplicationThread 接口类型,具体实现是 ActivityThread 的内部类 ApplicationThread。因此,后续执行 Activity 生命周期的过程都是由 ApplicationThread 指导完成的。

上面代码中的 scheduleTransaction() 方法仍然是调用了 ActivityThread 的 scheduleTransaction() 方法,但实际上这个方法是在 ActivityThread 的父类 ClientTransactionHandler 中实现。

 调用 sendMessage() 方法向 Hanlder 中发送了一个 EXECUTE_TRANSACTION 消息,并且 Message 中的 obj 就是启动 Activity 的事务对象。而这个 Handler 的具体实现是 ActivityThread 中的 mH 对象。具体代码如下

最终调用了事务的 execute() 方法,如下代码所示

在 executeCallback 方法中,会遍历事务中的 callback 并执行 execute 方法。这些 callbacks 是何时被添加的呢?

前面在创建 ClientTransaction 时,通过 addCallback() 方法传入了callback 参数。从下面代码可以看出,其实是一个 LaunchActivityItem 对象

LaunchActivityItem 的 execute() 方法

这里就已经到了与 Activity 生命周期相关的方法。上面代码中的 client 是 ClientTransactionHandler 类型,实际实现类是 ActivityThread,因此最终方法又回到了 ActivityThread。

ActivityThread 的 handleLaunchActivity

这是一个比较重要的方法,Activity 的生命周期方法就是在这个方法中有序执行。

图中1处,初始化 Activity 的 WindowManager,每一个 Activity 都会对应一个窗口;图中2处,调用 performLaunchActivity 创建并显示 Activity;图中3处通过反射创建目标 Activity 对象;图中4处调用 attach 方法建立 activity 与 context 之间的联系。创建 phoneWindow 对象,并与 Activity 进行关联操作;图中5处通过 instrumentation 最终调用 Activity 的 onCreate 方法。

至此,目标 Activity 已经成功创建,并执行生命周期方法。

总结

本次详细介绍了 Activity 的启动在源码中的实现流程

这一过程主要涉及 3 个进程间的通信过程:

\bullet 进程 A 通过 Binder 调用 AMS 的 startActivity 方法;

\bullet AMS 通过一系列的计算构造目标 Intent,然后在 ActivityStack 与 ActivityStackSupervisor 中处理 Task 和 Activity 的入栈操作;

\bullet AMS 通过 Binder 机制,调用目标进程中 ApplicationThread 的方法来创建并执行 Activity 生命周期方法,ApplicationThread 是 ActivityThread 的一个内部类,最终都调用到了 ActivityThread 中的相应方法。

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

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

相关文章

美食论坛大全订阅交流系统 uniapp+vue 微信小程序设计与实现

美食大全订阅小程序在系统的智能手机上可以运行,主要实现了首页、个人中心、用户管理、菜系管理、口味管理、美食分类管理、美食信息管理、美食论坛、系统管理等主要功能模块的操作和管理。 后端:java(springbootssm)/python/php/nodejs/ 开发运行&…

C++ stack queue 的模拟实现

1.为什么选择 deque 作为 stack 和 queue 的默认容器呢? stack 是一种后进先出的特殊线性数据结构,因此只要具有 push_back() 和 pop_back() 操作的线性结构,都可 以作为 stack 的底层容器,比如 vector 和 list 都可以&#xff1b…

ch3_6多线程举例

作者丨billom 来源丨投稿 编辑丨GiantPandaCV 云端深度学习的服务的性能加速通常需要算法和工程的协同加速,需要模型推理和计算节点的融合,并保证整个“木桶”没有太明显的短板。 如何在满足时延前提下让算法工程师的服务的吞吐尽可能高,尽…

【论文阅读VLDB23】Online Schema Evolution is (Almost) Free for Snapshot Databases

Online Schema Evolution is (Almost) Free for Snapshot Databases 对snapshot DB来说可以几乎在线进行schema的演变。 ABSTRACT 在现有数据库系统中,对在线和事务模式演变的支持仍然具有挑战性。之前处理这种schema 演变基本是patches补丁的方式来做&#xff0…

【卷积神经网络】YOLO 算法原理

在计算机视觉领域中,目标检测(Object Detection)是一个具有挑战性且重要的新兴研究方向。目标检测不仅要预测图片中是否包含待检测的目标,还需要在图片中指出它们的位置。2015 年,Joseph Redmon, Santosh Divvala 等人…

物联网AI MicroPython传感器学习 之 mpu6050六轴陀螺仪传感器

学物联网,来万物简单IoT物联网!! 一、产品简介 MPU6050是一款6轴运动传感器,它集成了3 轴MEMS 陀螺仪,3 轴MEMS加速度计,以及一个可扩展的数字运动处理器DMP(Digital Motion Processor&#xf…

moviepy处理手机端图片旋转问题

1.手机拍摄的图片或者在ffmpeg处理的时候,会读取图片的元数据从而进行旋转 左边是拍摄的图片,右侧是进行处理以后得图片 video VideoFileClip(file_path) if video.rotation in (90, 270):video video.resize(video.size[::-1])video.rotation 0

YouTube博主数据信息资源

YouTube博主数据信息资源 🔥我是一位拥有10年编程经验的程序猿,为你带来一个全新的优质资源 🔍您是否在寻找最新、最活跃的YouTube博主数据,以助力你的项目、营销或研究? 我们的数据,您的优势:…

ZYNQ连载05-Vitis更新xsa硬件配置

ZYNQ连载05-Vitis更新xsa硬件配置 1. 简述 在开发过程中,Vivado中硬件配置在开发过程中有所变动,Vitis需要根据Vivado生成的xsa文件,更新相应的BSP配置 2. 操作步骤

Node编写更新用户信息接口

目录 前言 定义路由和处理函数 验证表单数据 实现更新用户基本信息的功能 前言 继前面几篇文章,本文介绍如何编写更新用户信息接口 定义路由和处理函数 路由 // 更新用户信息接口 router.post(/userinfo, userinfo_handler.updateUserinfo) 处理函数 // 导…

【2023年冬季】华为OD统一考试(B卷)题库清单(已收录345题),又快又全的 B 卷题库大整理

目录 专栏导读华为OD机试算法题太多了,知识点繁杂,如何刷题更有效率呢? 一、逻辑分析二、数据结构1、线性表① 数组② 双指针 2、map与list3、队列4、滑动窗口5、二叉树6、并查集7、栈 三、算法1、基础算法① 贪心算法② 二分查找③ 分治递归…

【PC电脑windows-学习样例generic_gpio-ESP32的GPIO程序-基础样例学习】

【PC电脑windows-学习样例generic_gpio-ESP32的GPIO程序-基础样例学习】 1、概述2、实验环境3、 物品说明4、自我总结5、本次实验说明6、实验过程(1)复制目录到桌面(2)手动敲写(3)反复改错(4&am…

时序预测 | Python实现ARIMA-LSTM差分自回归移动平均模型结合长短期记忆神经网络时间序列预测

时序预测 | Python实现ARIMA-LSTM差分自回归移动平均模型结合长短期记忆神经网络时间序列预测 目录 时序预测 | Python实现ARIMA-LSTM差分自回归移动平均模型结合长短期记忆神经网络时间序列预测预测效果基本介绍程序设计参考资料 预测效果 基本介绍 时序预测 | Python实现ARIM…

SOLIDWORKS PDM 2024数据管理5大新功能

1. 改进的视觉内容 • 通过装配体可视化功能,在 SOLIDWORKS 中以图形方式查看零部件数据,如工作流程状态。• 使用特定图标迅速识别焊件切割清单零部件。 优点:重要数据和系统信息一目了然。 2.增强的数据保护和跟踪功能 •保护“包含”和…

代码随想录算法训练营第4天| 24. 两两交换链表中的节点、19.删除链表的倒数第N个节点、面试题 02.07. 链表相交 、142.环形链表II

JAVA语言编写 24. 两两交换链表中的节点 谷歌、亚马逊、字节、奥多比、百度 给你一个链表,两两交换其中相邻的节点,并返回交换后链表的头节点。你必须在不修改节点内部的值的情况下完成本题(即,只能进行节点交换)。…

Openssl数据安全传输平台010:jasoncpp 1.9.5编译及常用API- Windows/Centos8-含测试代码

文章目录 0. 代码仓库1 安装注意版本问题1.1 windows 下的安装1.1.1 使用cmake生成sln工程1.1.2 使用sln工程生成静态库 -Debug 1.2 Linux 下的安装1.2.1 安装jsoncpp1.2.2 旧版本0.10.x版本相关环境配置问题 2 Winsows环境下编译 c 测试文件: json-test.cpp2.1 配置…

nginx请求时找路径问题

nginx请求时找路径问题 你是否遇到过这样的情况: 当你安装了nginx的时候,为nginx配置了如下的location,想要去访问路径下面的内容,可是总是出现404,找不到文件,这是什么原因呢,今天我们就来解…

服务器数据恢复—nas硬盘故障导致raid6失效、存储无法访问的数据恢复案例

服务器故障&分析: 一台nas存储中有一组由十几块硬盘组建的raid6磁盘阵列。 nas存储中的raid6阵列成员盘出现故障离线,磁盘阵列崩溃,nas存储无法正常访问。 北亚企安数据恢复工程师将nas存储内的所有硬盘编号后取出,经过硬件工…

MGRE环境下的OSPF

实验拓扑 需求 1 R6为ISP只能配置IP地址,R1-R5的环回为私有网段 2 R1/4/5为全连的MGRE结构,R1/2/3为星型的拓扑结构,R1为中心站点 3 所有私有网段可以互相通讯,私有网段使用OSPF完成。 IP规划 配置IP R1 # interface GigabitEt…