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 中的相应方法。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

竞赛选题 深度学习人体跌倒检测 -yolo 机器视觉 opencv python

0 前言 🔥 优质竞赛项目系列,今天要分享的是 🚩 **基于深度学习的人体跌倒检测算法研究与实现 ** 该项目较为新颖,适合作为竞赛课题方向,学长非常推荐! 🥇学长这里给一个题目综合评分(每项满…

STM32-ADC实验

目录 实验1:单ADC单通道中断 硬件原理图 USART配置 ADC1配置 初始化结构体的参数 ScanConvMode:扫描转换模式 ContinuousConvMode:连续转换模式 ExternalTrigConv:外部触发方式 测试环节 实验现象 实验2:单…

【疑问解决】在自动装箱中Integer赋予一个常量1,为什么会出现==判断true和flase的情况(JDK源码、内部缓冲)

问题来源自讲课时的Integer练习中 当时第一反应是false true true 因为第一段的输出为flase毋庸置疑了,因为已经new了两个新的堆空间,当然指向不同的空间了 但是第二段第三段就没有头绪了,自动装箱了难道不是执行同一个空间吗…

服务网络基础

服务网络基础 目录 前言 从今天开始我们将进入服务网格的学习,服务网格是微服务架构中的一种重要的技术,它可以解决微服务架构中的一些问题,比如服务发现、服务治理、服务监控等等,我们将从服务网格的基础开始,逐步深…

怎么突破反爬虫机制

在当今的数字化时代,网络爬虫已经成为了收集信息和数据的重要工具。然而,许多网站和平台都配备了反爬虫机制,以防止恶意攻击和过度访问。对于普通用户来说,如何突破这些反爬虫机制呢?本文将为你提供一些实用的技巧和建…

权限系统设计(转载)

1 为什么需要权限管理 2 权限模型 2.1 权限设计 2.2 为什么需要角色 2.3 权限模型的演进 2.4 用户划分 2.5 理想的RBAC模型 3 权限系统表设计 3.1 标准RBAC模型表设计 3.2 理想RBAC模型表设计 4 结语 1 为什么需要权限管理 日常工作中权限的问题时时刻刻伴随着我们&a…

流程引擎-自定义函数的应用

背景: 某些业务需求比较特殊,需要在表单中校验或实现一些功能,泛微流程表单配置时实现的方式多种多样:JS脚本、SQL语句、公式以及其他一些标准化拖拽功能,本次给大家分享一下流程表单中的公式实现的一些需求场景。泛微…

Python轮廓追踪【OpenCV形态学操作】

文章目录 概要代码运行结果 概要 一些理论知识 OpenCV形态学操作理论1 OpenCV形态学操作理论2 OpenCV轮廓操作|轮廓类似详解 代码 代码如下,可以直接运行 import cv2 as cv# 定义结构元素 kernel cv.getStructuringElement(cv.MORPH_RECT, (3, 3)) # print kern…

安装虚拟机找不到虚拟网啦1(eth1)不出现

一、安装虚拟机找不到虚拟网啦1(eth1)不出现 1、先安装virtualbox 2、再安装vagrant 3、在windows使用ipconfig没有VirtualBoxHost-OnlyNetWork解决方法 1) 解决办法 在windows的设置中找到 网络和Internet 选项,选择右侧 更改适配器选项 …

selenium+python web自动化测试框架项目实战实例教程

自动化测试对程序的回归测试更方便。 由于回归测试的动作和用例是完全设计好的,测试期望的结果也是完全可以预料的,将回归测试自动运行... 可以运行更加繁琐的测试 自动化测试的一个明显好处就是可以在很短的时间内运行更多的测试。学习自动化测试最终目的是应用到实际项目中&…

Android Studio 导出 jar

AS版本:Android Studio Giraffe | 2022.3.1 Patch 1 1、File——New Module——Android Library 2、mylibrary——main——新建功能类 3、mylibrary——build.gradle——android {}内复制以下代码——Sync Now //Copy类型 tasks.register(makeJar, Copy) { //删…

【Java 进阶篇】Java Request 继承体系详解

在Java编程中,Request(请求)是一个常见的概念,特别是在Web开发中。Request通常用于获取来自客户端的信息,以便服务器能够根据客户端的需求提供相应的响应。在Java中,Request通常涉及到一系列类和接口&#…

CentOS 编译安装TinyXml2

安装 TinyXml2 Git 源码下载地址:https://github.com/leethomason/tinyxml2 步骤1:首先,你需要下载tinyxml2的源代码。你可以从Github或者源代码官方网站下载。并上传至/usr/local/source_code/ 步骤2:下载完成后,需要将源代码解…

Java IDEA feign调用上传文件MultipartFile以及实体对象亲测可行

Java IDEA feign调用上传文件MultipartFile以及实体对象亲测可行 1. 报错 java.lang.IllegalStateException: Body parameter cannot be used with form parameters2. 解决参考 1. 报错 java.lang.IllegalStateException: Body parameter cannot be used with form parameters …

1-多媒体通信概述

文章目录 媒体和多媒体媒体多媒体VarityIntergrationInteraction 多媒体通信(MMC)业务类型 MMC主要问题和关键技术主要问题关键技术 MMC发展动向重要事件趋势 标准化组织 媒体和多媒体 媒体 承载信息的载体. 感知媒体, 表示媒体, 显示媒体, 存储媒体, 传输媒体. 多媒体 Var…

【设计模式】第3节:设计模式概论

设计模式不是代码,而是某类问题的通用方案。设计模式的本质是提高软件的维护性、通用性和扩展性,并降低软件的复杂度。一共有24种设计模式,可以分为创建型模式、结构型模式和行为型模式三大类。设计模式中比较重要的有:单例模式、…

Spring Authorization Server 1.1 扩展实现 OAuth2 密码模式与 Spring Cloud 的整合实战

目录 前言无图无真相创建数据库授权服务器maven 依赖application.yml授权服务器配置AuthorizationServierConfigDefaultSecutiryConfig 密码模式扩展PasswordAuthenticationTokenPasswordAuthenticationConverterPasswordAuthenticationProvider JWT 自定义字段自定义认证响应认…

C++(Qt)软件调试---线程死锁调试(15)

C(Qt)软件调试—线程死锁调试(15) 文章目录 C(Qt)软件调试---线程死锁调试(15)1、前言2、常见死锁3、linux下gdb调试C死锁1.1 使用代码1.2 gdb调试 3、linux下gdb调试Qt死锁1.1 使用代码1.2 gdb调试 4、Windows下gdb调试C死锁5、W…

Selenium+Pytest自动化测试框架详解

前言 selenium自动化 pytest测试框架 本章你需要 一定的python基础——至少明白类与对象,封装继承;一定的selenium基础——本篇不讲selenium,不会的可以自己去看selenium中文翻译网 一、测试框架简介 测试框架有什么优点 代码复用率高&…

【LeetCode】5. 最长回文子串

题目链接 文章目录 Python3方法: 暴力求解 ⟮ O ( n 3 ) 、 O ( 1 ) ⟯ \lgroup O(n^3)、O(1)\rgroup ⟮O(n3)、O(1)⟯方法一: 动态规划 (回文串同时去掉头尾后 依然是回文串) ⟮ O ( n 2 ) ⟯ \lgroup O(n^2)\rgroup ⟮O(n2)⟯⭐ 方法二:…