再谈启动一个Activity大致时序图

news2025/2/4 15:57:22

太多了,笔者不想写,

读者可通过PlantUML插件查看如下PUML文件生成的时序图。

补充说明下,Android31版本。

@startuml
'https://plantuml.com/sequence-diagram
skinparam dpi 800
scale 15000 width
scale 5000 height

autonumber
==Launcher==
Launcher -> Activity:startActivity(intent)
Activity->Activity:startActivity(intent,bundle)
Activity->Activity:startActivityForResult
Activity->Instrumentation:execStartActivity
note left
activity通过调用内部成员
mInstrumentation与atms
交互,即将开始进入系统进程
end note
Instrumentation->ActivityTaskManagerService.Stub:startActivity
==SystemService==
ActivityTaskManagerService.Stub->ActivityTaskManagerService:StartActivity
ActivityTaskManagerService->ActivityTaskManagerService:startActivityAsUser
ActivityTaskManagerService->ActivityTaskManagerService:startActivityAsUser
ActivityTaskManagerService->ActivityStarter:execute
ActivityStarter->ActivityStarter:executeRequest
note left
这里,会验证启动参数的合法性,
随后,创建一个ActivityRecord
作为一个Activity的唯一存在标识
end note
ActivityStarter->ActivityStarter:startActivityUnchecked
ActivityStarter->ActivityStarter:startActivityInner
note left
此函数的功能,如果没有Stack,就要新创建一个Stack,
也就是ActivityStack,保存到mTargetTask中,
通过setNewTask将待启动的ActivityRecord
和新创建的Stack绑定,并且将其置于Stack Top,
see ActivityStack::startActivityLocked()
see ActivityStack::moveToFront()
end note
ActivityStarter->ActivityStack:startActivityLocked
note left
这就是绑定ActivityRecord和新创建的Stack(Task)
end note

ActivityStack->RootWindowContainer:resumeFocusedStacksTopActivities
note left
对于新启动的Activity,走此分支
end note

RootWindowContainer->ActivityStack:resumeTopActivityUncheckedLocked
note left
此ActivityStack是13步新创建的,
开始调度Activity
end note

ActivityStack->ActivityStack:resumeTopActivityInnerLocked

note left
这一步很关键,因为要Resume即将启动的Activity,
那么,首先要检查即将启动的Activity在没在栈顶,
如果不在,就要去pause栈顶的Activity
end note
ActivityStack->TaskDisplayArea:pauseBackStacks
note left
这里返回是否在pause栈顶的Activity,
一般这里会返回true,因为需要pause Launcher,
传入next参数即topRunningActivity
end note

note right
逻辑是去topStack中去判断是否有可见Activity,
如果有则pause掉,否则返回false,
end note
TaskDisplayArea->ActivityStack:startPausingLocked
note right
开始pause Launcher,
通过mAtmService(ATMS)获取生命周期管理,
执行生命周期PauseActivityItem事务
end note
ActivityStack->ClientLifecycleManager:scheduleTransaction

note left
注意,这里传的参数有PauseActivityItem
end note
ClientLifecycleManager->ClientLifecycleManager:scheduleTransaction
ClientLifecycleManager->ClientTransaction:schedule
ClientTransaction->IApplicationThread:scheduleTransaction
note right
进入Launcher进程,去执行Pause事务
end note

TaskDisplayArea-->ActivityStack:pauseBackStacks:true
ActivityStack-->RootWindowContainer:resumeTopActivityInnerLocked:true
RootWindowContainer-->ActivityStarter:resumeTopActivityUncheckedLocked:true
ActivityStarter-->ActivityStarter:ActivityStarter:START_SUCCESS


==Launcher进程==
IApplicationThread->ActivityThread:scheduleTransaction
ActivityThread->ActivityThread:sendMessage(159,transaction)
ActivityThread->Handler:sendMessage
note right
这里159代表事务EXECUTE_TRANSACTION
一顿Looper后可以直接看handleMessage处
end note

Handler->Handler:handleMessage
ActivityThread->TransactionExecutor:execute

TransactionExecutor->TransactionExecutor:executeLifecycleState
TransactionExecutor->PauseActivityItem:execute
PauseActivityItem->ActivityThread:handlePauseActivity
note right
熟悉的ActivityThread,开始pause Activity
end note

ActivityThread->ActivityThread:performPauseActivity
ActivityThread->ActivityThread:performPauseActivityIfNeeded
ActivityThread->Instrumentation:callActivityOnPause
Instrumentation->Activity:performPause
Activity->Activity:onPause

TransactionExecutor->PauseActivityItem:postExecute
PauseActivityItem->ActivityTaskManagerService.Stub:activityPaused
note left
通知系统进程,Launcher已经pause
end note
==SystemService==
ActivityTaskManagerService->ActivityRecord:activityPaused(false/*timeout*/)
note right
拓展:这里就是其中一种ANR,如果没超时就移除anr回调,
end note
ActivityRecord->ActivityStack:completePauseLocked(true/*resumeNext*/,null)
ActivityStack->RootWindowContainer:resumeFocusedStacksTopActivities

RootWindowContainer->ActivityStack:resumeTopActivityUncheckedLocked

note right
找到下一个即将启动的ActivityStack,开始Resume
end note

ActivityStack->ActivityStack:resumeTopActivityInnerLocked

ActivityStack->TaskDisplayArea:pauseBackStacks
TaskDisplayArea-->ActivityStack:pauseBackStacks:false

ActivityStack->ActivityStackSupervisor:startSpecificActivity
note right
这里通过ActivityRecord去获取WindowProcessController,
这个和进程相关,第一次启动肯定没有对应进程,那么就需要创建进程
end note

ActivityStackSupervisor->ActivityTaskManagerService:startProcessAsync
ActivityTaskManagerService->ActivityManagerInternal:startProcess
note right
这里通过handler调用到此处,
ActivityManagerInternal是一个抽象类,
其此处实例是ActivityServiceManager的LocalService
end note
ActivityManagerInternal->ActivityManagerService:startProcessLocked
ActivityManagerService->ProcessList:startProcessLocked
ProcessList->ProcessList:startProcessLocked
ProcessList->ProcessList:startProcessLocked
ProcessList->ProcessList:startProcessLocked
ProcessList->ProcessList:startProcess
ProcessList->Process:start
Process->ZygoteProcess:start
ZygoteProcess->ZygoteProcess:startViaZygote
note right
写入参数,通过socket发送给Zygote
end note
ZygoteProcess->ZygoteProcess:zygoteSendArgsAndGetResult
ZygoteProcess->ZygoteProcess:attemptZygoteSendArgsAndGetResult
ZygoteProcess->ZygoteState:write
note right
正式发送给zygote
end note
ZygoteProcess->ZygoteState:read
note right
主动读取结果,获取新启动的进程pid,保存到上下文中,
等待attach新启动的进程,由此,我们进入Zygote进程
end note

==Zygote==
ZygoteInit->ZygoteInit:main
ZygoteInit->ZygoteServer:runSelectLoop
ZygoteServer->ZygoteConnection:processOneCommand
ZygoteConnection->Zygote:forkAndSpecialize
note right
这个返回值很关键,如果在此处将fork出两个进程,
如果是新创建的进程,pid为0,否则就是Zygote进程
end note

Zygote-->ZygoteConnection:pid
ZygoteConnection->ZygoteConnection:handleParentProc
ZygoteConnection->ZygoteState:write
note right
写入新创建的pid给SystemService
end note
ZygoteConnection-->ZygoteServer:processOneCommand:Runnable
ZygoteServer->ZygoteServer:runSelectLoop
note right
继续监听其它连接,完毕
end note
==App进程==
Zygote-->ZygoteConnection:pid
ZygoteConnection->ZygoteConnection:handleChildProc
ZygoteConnection->closeSocket
ZygoteConnection->ZygoteInit:childZygoteInit
ZygoteInit->RuntimeInit:findStaticMain
note right
这里找到ActivityThread.main()方法
end note
RuntimeInit-->ZygoteInit:main
RuntimeInit-->ZygoteConnection:main
ZygoteConnection-->ZygoteServer:main
ZygoteServer-->ZygoteInit:runSelectLoop:main
ZygoteInit->ActivityThread:main
note left
至此,app的入口正式启动,
这里负责创建Looper.prepare();
还有最重要的attach,毕竟刚创建的app是个新生儿,
需要把自己注册到android世界中,才能获得活动的权利
最后进入主线程死循环loop
end note
ActivityThread->ActivityThread:attach
ActivityThread->ActivityManagerService.Stub:attachApplication
==SystemService==
ActivityManagerService.Stub->ActivityManagerService:attachApplication
ActivityManagerService->ActivityManagerService:attachApplicationLocked
ActivityManagerService->ActivityTaskManagerInternal
note right
这个是ATMS的LocalService
end note
ActivityTaskManagerInternal->RootWindowContainer:attachApplication
RootWindowContainer->RootWindowContainer:startActivityForAttachedApplicationIfNeeded
RootWindowContainer->ActivityStackSupervisor:realStartActivityLocked
note left
现在进程有了,该真正启动了
end note
ActivityStackSupervisor->ClientLifecycleManager:scheduleTransaction
note right
这里面约莫842行,给事务添加了回调,LaunchActivityItem;
然后也设置的生命周期状态ResumeActivityItem
end note
ClientLifecycleManager->ClientTransaction:schedule
ClientTransaction->IApplicationThread:scheduleTransaction
==App进程==
IApplicationThread->ActivityThread:scheduleTransaction
ActivityThread->ActivityThread:sendMessage(159,transaction)
ActivityThread->Handler:sendMessage
Handler->TransactionExecutor:execute
TransactionExecutor->TransactionExecutor:executeCallbacks
note right
注意此处的LaunchActivityItem
end note
TransactionExecutor->LaunchActivityItem:execute
LaunchActivityItem->ActivityThread:handleLaunchActivity
ActivityThread->ActivityThread:performLaunchActivity
ActivityThread->Instrumentation:newActivity
Instrumentation-->ActivityThread:activity
ActivityThread->Activity:attach

note left
那么,activity就被反射创建了,
这里,创建了PhoneWindow等必要成员
end note
ActivityThread->Instrumentation:callActivityOnCreate
Instrumentation->Activity:performCreate
Activity->Activity:onCreate
Activity-->TransactionExecutor
TransactionExecutor->TransactionExecutor:executeLifecycleState
TransactionExecutor->TransactionExecutor:cycleToPath
note left
这样终态传入的是Resume,因此需要在此处拿到IntArray的生命周期表
end note
TransactionExecutor->TransactionExecutor:performLifecycleSequence
note left
挨个走start、resume,
end note
TransactionExecutor->ActivityThread:handleStartActivity
ActivityThread->Activity:performStart
Activity->Instrumentation:callActivityOnStart
Instrumentation->Activity:onStart
Activity-->TransactionExecutor

TransactionExecutor->ActivityThread:handleResumeActivity
ActivityThread->Activity:performResume
Activity->Instrumentation:callActivityOnResume
Instrumentation->Activity:onResume
Activity-->ActivityThread:handleResumeActivity
note left
此处,继续走handlerResumeActivity接下来的分支,
就是绑定Activity和Window的过程,
wm.addView(decor, l);
自此就是View绘制流程(极多的另一流程),然后显示在屏幕上
end note

Activity-->ActivityThread:main
ActivityThread->Looper:loop
@enduml

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

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

相关文章

AJAX-入门

定义 概念:AJAX是浏览器与服务器进行数据通信的技术 使用 1.先使用axios库,与服务器进行数据通信 1)基于XMLHttpRequest封装、代码简单、月下载量在14亿次 2)Vue、React项目中都会用到axios 2.再学习XMLHttpRequest对象的使用…

学习日志以及个人总结 (16)

共用体 共用体 union 共用体名 { 成员列表; };//表示定义一个共用体类型 注意: 1.共用体 初始化 --- 只能给一个值,默认是给到第一个成员变量 2.共用体成员变量辅助 3.可以判断大小端 ----※!! 实际用途…

Flask框架开发学习笔记《5》简易服务器代码

Flask框架开发学习笔记《5》 Flask是使用python的后端,由于小程序需要后端开发,遂学习一下后端开发。 简易服务器代码 接口解析那一块很关键,学后端服务器这一块,感觉主要就是学习相应地址的接口怎么处理。 然后写清楚每个地址…

第96讲:MySQL高可用集群MHA的核心概念以及集群搭建

文章目录 1.MHA高可用数据库集群的核心概念1.1.主从复制架构的演变1.2.MHA简介以及架构1.3.MHA的软件结构1.4.MHA Manager组件的启动过程1.5.MHA高可用集群的原理 2.搭建MHA高可用数据库集群2.1.环境架构简介2.2.搭建基于GTID的主从复制集群2.2.1.在三台服务器中分别搭建MySQL实…

C#验证字符串是正整数还是负整数,正则表达式vs用Char.IsDigit 方法遍历字符数组

目录 一、使用的方法 1.正则表达式 2.Char.IsDigit 方法 二、源码 1.源代码 2.生成效果 一、使用的方法 1.正则表达式 使用正则表达式Regex类的IsMatch方法,可以有效地判断用户输入的信息是否为有符号整数。 用于判断字符串是否有符号整数的正则表达式…

Maya------显示隐藏提取复制刺破面

alth<--->ctrlshifth 补洞后刺破面&#xff0c;防止多边面的产生&#xff01;

【C/C++ 09】万年历

一、题目 输入一个年份&#xff0c;以日历的格式打印这一年的所有天数&#xff0c;需要正确的表示每一天是周几。 二、算法 以公元1年1月1日作为万年历的起始日期&#xff0c;公元1年1月1日是周一&#xff0c;所以算法的核心就是就算某一天距离起始日期的天数差&#xff0c;然…

Android14之Selinux报错:unknown type qemu_device at token (一百八十三)

简介&#xff1a; CSDN博客专家&#xff0c;专注Android/Linux系统&#xff0c;分享多mic语音方案、音视频、编解码等技术&#xff0c;与大家一起成长&#xff01; 优质专栏&#xff1a;Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 优质专栏&#xff1a;多媒…

17.STL 库(C++)plus

STL 库&#xff08;C&#xff09; 文章目录 STL 库&#xff08;C&#xff09;1.迭代器1.1 概述和分类1.2案例 2.C 的 string 类型2.1string 概述2.2 string 构造函数2.3 string 赋值操作2.4string存取字符串操作2.5string拼接操作2.6 string 查找和替换2.7string 比较操作2.8st…

AJAX-axios错误处理

场景&#xff1a; 处理&#xff1a;用更直观的方式&#xff0c;给普通用户展示错误信息 语法&#xff1a;在then方法的后面&#xff0c;通过点语法调用catch方法&#xff0c;传入回调函数并定义形参 axios({//请求选项}).then(result> {//处理数据}).catch(error>{//处理…

Qt 范例阅读: QStateMachine状态机框架 和 SCXML 引擎简单记录(方便后续有需求能想到这两个东西)

一、QStateMachine 简单应用&#xff1a; 实现按钮的文本切换 QStateMachine machine; //定义状态机&#xff08;头文件定义&#xff09;QState *off new QState(); //添加off 状态off->assignProperty(ui->pushButton_2, "text", "Off"); //绑定该…

2023_12蓝桥杯STEMA 考试 Scratch 中级试卷解析

2023蓝桥杯STEMA 考试 Scratch 中级试卷(12 月)解析 由于没有原始文件,这里使用的角色和背景和实际题目会有所差异,已经尽量还原原题,以下代码仅供参考。吐槽一句:蓝桥杯越来越变态了!\(`Δ’)/\(`Δ’)/\(`Δ’)/孩子学习速度永远也赶不上内卷的速度。 一、选择…

【axios报错异常】: Uncaught ReferenceError: axios is not defined

问题描述: 当前代码在vivo手机和小米手机运行是正常的,点击分享按钮调出相关弹框,发送接口进行分享,但是现在oppo手机出现了问题: 点击分享按钮没有反应. 问题解析: 安卓同事经过查询后,发现打印了错误: 但是不清楚这个问题是安卓端造成的还是前端造成的,大家都不清楚. 问题…

MySQL系列:系列结构和基础管理

文章目录 MySQL工作模型及实例MySQL 客户端/服务器工作模型&#xff08;C/S&#xff09;服务端&#xff1a;实例MySQLd的程序结构MySQL的逻辑结构MySQL的物理存储结构 MySQL基础管理用户管理权限管理连接管理初始化配置启动关闭多实例 MySQL工作模型及实例 MySQL 客户端/服务器…

【线上研讨会】PowerFLOW汽车空调噪声和风扇噪声解决方案

会议地址&#xff1a; 达索系统官方在线研讨会平台

2024美赛E题保姆级分析完整思路代码数据教学

2024美国大学生数学建模竞赛E题保姆级分析完整思路代码数据教学 E题&#xff1a;Sustainability of Property Insurance&#xff08;财产保险的可持续性&#xff09; 这道题目同样是比赛的热门题目&#xff0c;是很多同学在训练的时候经常做的题目类型了&#xff0c;属于大数据…

【Qt】—— Hello World程序的实现

目录 &#xff08;一&#xff09;使⽤"按钮"实现 1.1 纯代码方式实现 1.2 可视化操作实现 &#xff08;二&#xff09;使⽤"标签"实现 2.1 纯代码方式实现 2.2 可视化操作实现 &#xff08;一&#xff09;使⽤"按钮"实现 1.1 纯代码方式实…

python爬虫3

1.异常处理&#xff0c;使代码更加健壮 静态cookie可视绕过登录的限制 快代理是一个代理平台 # https://movie.douban.com/j/chart/top_list?type5&interval_id100%3A90&action& # start0&limit20# https://movie.douban.com/j/chart/top_list?type5&int…

Java_简单模拟实现ArrayList_学习ArrayList

文章目录 一、 了解线性表和顺序表区别1.线性表2.顺序表 二、模拟实现1.定义接口2.定义MyArrayList3.成员变量以及构造方法4.实现打印数组5.实现add方法6.实现查找某个数是否存在contains或者某个数的下标indexOf7.获取或更改pos位置的值 get和set8.获取数组大小 size9.删除某个…

C语言:文件操作详解

创作不易&#xff0c;友友们给个三连吧&#xff01;&#xff01; 一、为什么我们需要使用文件 我们在写程序的时候&#xff0c;输入的数据是存储在电脑内存中的&#xff0c;如果程序退出内存回收&#xff0c;相应数据也就丢失了&#xff0c;等再次运行程序&#xff0c;就看不到…