内容简介:应用程序的启动,又可称为根Activity的启动。但是在讲应用程序启动之前,我们有必要对应用程序进程(AppProcess)启动有所了解,那是因为启动一个应用程序首先要保证该应用程序的进程已经被启动。AMS在启动应用程序时,会先检查应用程序进程是否存在,如果不存在就需要请求Zygote进程创建并启动应用程序进程。这里我不会贴上大段大段的代码,只是一些总结,并提供了相关源码的链接。讲完了应用程序进程(AppProcess)启动的相关内容后,接下来我们就来看看应用程序是如何一步一步启动的。
应用程序的启动,又可称为根Activity的启动。但是在讲应用程序启动之前,我们有必要对应用程序进程(AppProcess)启动有所了解,那是因为启动一个应用程序首先要保证该应用程序的进程已经被启动。AMS在启动应用程序时,会先检查应用程序进程是否存在,如果不存在就需要请求Zygote进程创建并启动应用程序进程。这里我不会贴上大段大段的代码,只是一些总结,并提供了相关源码的链接。
应用程序进程(AppProcess)启动
启动大纲
- AMS发送启动应用程序进程请求.
- Zygote接收请求并创建应用程序进程.
AMS发送启动应用程序进程请求
-
AMS通过调用startProcessLocked方法向Zygote进程发送请求。
-
Process 调用
start
方法,使用ZygoteProcess的start
方法。 -
在ZygoteProcess的
start
方法中,先后调用了startViaZygote
、zygoteSendArgsAndGetResult
和openZygoteSocketIfNeeded
等方法,最后在openZygoteSocketIfNeeded
方法中调用了ZygoteState的connect方法建立与Zygote进程的连接。
AMS发送启动应用程序进程请求
-
ZygoteServer 执行
runSelectLoop
方法,一直等待AMS的请求数据到来。 -
当AMS请求到来,与Zygote进程建立连接后,由ZygoteConnection的
processOneCommand
方法处理请求的数据。对请求数据进行解析,获取程序进程的启动参数,并通过Zygote的forkAndSpecialize
方法进行应用程序进程的创建。 -
进程创建完成后,交由ZygoteInit的
zygoteInit
方法和RuntimeInit的applicationInit
方法分别进行进程和应用的初始化。在zygoteInit
方法中,为应用程序进程创建了Binder线程池,这样进程就可以跨进程进行通信了。而applicationInit
方法通过反射最终会调用ActivityThread的main
方法,从而完成应用程序进程的创建。
应用程序(App)启动
讲完了应用程序进程(AppProcess)启动的相关内容后,接下来我们就来看看应用程序是如何一步一步启动的。
启动大纲
-
Launcher请求AMS。
-
AMS请求ApplicationThread。
-
ActivityThread启动Activity。
启动时序图
Launcher请求AMS
-
当我们点击应用程序的图标时,就会自动调用Launcher的
startActivitySafely
方法, 最终会调用Activity的startActivity
方法。 -
在Activity的
startActivity
中又调用了startActivityForResult
方法,而startActivityForResult
方法内部又调用了Instrumentation的execStartActivity
方法。 -
在Instrumentation的
execStartActivity
方法中又通过ActivityManager的getService
方法获取了IBinder类型的AMS引用IActivityManager
,最后调用了AMS的startActivity
方法。
AMS请求ApplicationThread
-
在AMS的
startActivity
方法中,又调用了其本身的startActivityAsUser
方法,进行权限的检查。 -
权限检查完后,调用ActivityStarter的
startActivityMayWait
方法,并在该方法中解析处理应用程序需要的参数,并进行相关参数的初始化,最终会调用其startActivity
方法。而在startActivity
方法中又调用了startActivityUnchecked
方法来处理与栈管理相关的逻辑。 -
在处理完栈的关系后,紧接着会调用 ActivityStackSupervisor 的
resumeFocusedStackTopActivityLocked
方法获取需要启动的Activity所在栈的栈顶。 -
当需要启动的Activity的状态不是RESUMED状态,就需要调用ActivityStack的
resumeTopActivityUncheckedLocked
方法,而它的内部又调用了resumeTopActivityInnerLocked
方法进行一系列的栈状态的判断,最终又回调了ActivityStackSupervisor的startSpecificActivityLocked
方法。 -
在 ActivityStackSupervisor 的
startSpecificActivityLocked
方法中先是获取了即将启动的Activity所在的应用程序进程(就是在这个地方判断应用所在进程是否存在且已启动,如果没有启动,就需要启动应用程序进程),然后调用realStartActivityLocked
方法。 -
在 ActivityStackSupervisor 的
realStartActivityLocked
方法中,对启动的应用程序进程进行一系列的判断和处理,最终会调用IBinder类型的ApplicationThread引用IApplicationThread
,通过传入IApplicationThread
建立ClientTransaction
,加入执行LaunchActivityItem
任务,最终实现跨进程执行调用ActivityThread的handleLaunchActivity
方法。
ActivityThread启动Activity
-
在ActivityThread调用了它的
handleLaunchActivity
方法中,会先调用其performLaunchActivity
方法,之后调用handleResumeActivity
,将Activity的状态置为Resume。 -
在ActivityThread的 performLaunchActivity 方法中做了很多事情。
- 首先,执行了
createBaseContextForActivity
方法,创建要启动Activity的上下文; - 其次,调用执行了Instrumentation的
newActivity
方法来创建Activity实例; - 接着,调用LoadedApk的
makeApplication
方法,创建应用程序的Application; - 之后,调用需要启动的Activity的
attach
方法初始化Activity,创建Window对象并与Activity自身进行关联。 - 最后,调用执行了Instrumentation的
callActivityOnCreate
方法来启动Activity。
- 在Instrumentation执行了
callActivityOnCreate
方法中,会调用Activity的performCreate
方法,最终会调用Activity的onCreate
方法,这样应用程序也就启动了。
应用程序启动的进程关系图
应用程序启动的过程中,主要涉及了Launcher进程、SystemServer进程、Zygote进程和应用程序进程这四个进程,它们之间的关系如下。
以上所述就是小编给大家介绍的《Android源码分析--应用程序启动》,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对 码农网 的支持!
猜你喜欢:- Android里应用程序,应用程序窗口和视图对象之间的关系
- 使用 Bluemix、Watson Discovery 和 Cloudant 构建移动应用程序来分析其他应用程序
- ChromeOS 终端应用程序暗示其即将支持 Linux 应用
- 这几天在C程序中有哪些应用程序?
- Zookeeper详解-应用程序(七)
- Google官方应用程序架构指南
本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。
首席产品官2 从白领到金领
车马 / 机械工业出版社 / 79元
《首席产品官》共2册,旨在为产品新人成长为产品行家,产品白领成长为产品金领,最后成长为首席产品官(CPO)提供产品认知、能力体系、成长方法三个维度的全方位指导。 作者在互联网领域从业近20年,是中国早期的互联网产品经理,曾是周鸿祎旗下“3721”的产品经理,担任CPO和CEO多年。作者将自己多年来的产品经验体系化,锤炼出了“产品人的能力杠铃模型”(简称“杠铃模型”),简洁、直观、兼容性好、实......一起来看看 《首席产品官2 从白领到金领》 这本书的介绍吧!