腾讯精选面试重点问题:Android源码分析--应用程序启动

时间:2022-01-31 18:48:01

面试官:什么是应用程序启动

应用程序的启动,又可称为根Activity的启动。但是在讲应用程序启动之前,我们有必要对应用程序进程(AppProcess)启动有所了解,那是因为启动一个应用程序首先要保证该应用程序的进程已经被启动。AMS在启动应用程序时,会先检查应用程序进程是否存在,如果不存在就需要请求Zygote进程创建并启动应用程序进程。这里我不会贴上大段大段的代码,只是一些总结,并提供了相关源码的链接。

应用程序进程(AppProcess)启动

启动大纲

  1. AMS发送启动应用程序进程请求.
  2. Zygote接收请求并创建应用程序进程.

AMS发送启动应用程序进程请求

腾讯精选面试重点问题:Android源码分析--应用程序启动

  • AMS通过调用startProcessLocked方法向Zygote进程发送请求。

  • Process调用start方法,使用ZygoteProcess的start方法。

  • 在ZygoteProcess的start方法中,先后调用了startViaZygotezygoteSendArgsAndGetResultopenZygoteSocketIfNeeded等方法,最后在openZygoteSocketIfNeeded方法中调用了ZygoteState的connect方法建立与Zygote进程的连接。

AMS发送启动应用程序进程请求

腾讯精选面试重点问题:Android源码分析--应用程序启动

  • ZygoteServer执行runSelectLoop方法,一直等待AMS的请求数据到来。

  • 当AMS请求到来,与Zygote进程建立连接后,由ZygoteConnection的processOneCommand方法处理请求的数据。对请求数据进行解析,获取程序进程的启动参数,并通过Zygote的forkAndSpecialize方法进行应用程序进程的创建。

  • 进程创建完成后,交由ZygoteInit的zygoteInit方法和RuntimeInit的applicationInit方法分别进行进程和应用的初始化。在zygoteInit方法中,为应用程序进程创建了Binder线程池,这样进程就可以跨进程进行通信了。而applicationInit方法通过反射最终会调用ActivityThread的main方法,从而完成应用程序进程的创建。

    • *

应用程序(App)启动

讲完了应用程序进程(AppProcess)启动的相关内容后,接下来我们就来看看应用程序是如何一步一步启动的。

启动大纲

  1. Launcher请求AMS。
  2. AMS请求ApplicationThread。
  3. ActivityThread启动Activity。

启动时序图

腾讯精选面试重点问题:Android源码分析--应用程序启动

Launcher请求AMS

腾讯精选面试重点问题:Android源码分析--应用程序启动

  • 当我们点击应用程序的图标时,就会自动调用Launcher的startActivitySafely方法, 最终会调用Activity的startActivity方法。

  • 在Activity的startActivity中又调用了startActivityForResult方法,而startActivityForResult方法内部又调用了Instrumentation的execStartActivity方法。

  • 在Instrumentation的execStartActivity方法中又通过ActivityManager的getService方法获取了IBinder类型的AMS引用IActivityManager,最后调用了AMS的startActivity方法。

AMS请求ApplicationThread

腾讯精选面试重点问题:Android源码分析--应用程序启动

  • 在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

腾讯精选面试重点问题:Android源码分析--应用程序启动

  • 在ActivityThread调用了它的handleLaunchActivity方法中,会先调用其performLaunchActivity方法,之后调用handleResumeActivity,将Activity的状态置为Resume。

  • 在ActivityThread的performLaunchActivity方法中做了很多事情。
  1. 首先,执行了createBaseContextForActivity方法,创建要启动Activity的上下文;
  2. 其次,调用执行了Instrumentation的newActivity方法来创建Activity实例;
  3. 接着,调用LoadedApk的makeApplication方法,创建应用程序的Application;
  4. 之后,调用需要启动的Activity的attach方法初始化Activity,创建Window对象并与Activity自身进行关联。
  5. 最后,调用执行了Instrumentation的callActivityOnCreate方法来启动Activity。
  • 在Instrumentation执行了callActivityOnCreate方法中,会调用Activity的performCreate方法,最终会调用Activity的onCreate方法,这样应用程序也就启动了。

应用程序启动的进程关系图

应用程序启动的过程中,主要涉及了Launcher进程、SystemServer进程、Zygote进程和应用程序进程这四个进程,它们之间的关系如下。

腾讯精选面试重点问题:Android源码分析--应用程序启动

总结

好了,今天的分享就到这里,如果你对在面试中遇到的问题,或者刚毕业及工作几年迷茫不知道该如何准备面试并突破现状提升自己,对于自己的未来还不够了解不知道给如何规划,可以简信我获取一下相关面试题及答案资料包及学习视频。

专注分享大型Bat面试知识,后续会持续更新,希望通过这些高级面试题能够降低面试Android岗位的门槛,让更多的Android工程师理解Android系统,掌握Android系统。喜欢的话麻烦点击一个喜欢在关注一下~