这篇文章讲解当用户点击应用图标时,安卓如何启动你的应用。安卓系统做了很多幕后工作,来使得你的launch activity对用户可见。本文通过重要阶段的讲解和调用序列详细讲解这一过程。
安卓应用在这两个方面是独特的:
多个入口点:Android应用程序由不同的组件组成,它们可以调用其他应用程序拥有的组件。这些组件大致对应于任何应用程序的多个入口点。因此,它们不同于具有像main()方法那样的单个入口点的传统应用程序。
拥有自己的小世界:每个Android应用程序都生活在自己的世界中,它在单独的进程中运行,拥有自己的Dalvik VM实例,并分配有唯一的用户ID。
Android进程何时开始?
必要时会启动Android进程。
每当用户或其他系统组件请求执行属于您应用程序的组件(可能是服务,活动或意图接收器)时,Android系统都会为您的应用程序启动一个新进程(如果尚未运行)。通常,进程一直运行直到被系统杀死。应用程序流程是按需创建的,在您看到应用程序的启动活动启动并运行之前,发生了许多事情。
每个应用程序都在其自己的进程中运行:默认情况下,每个Android应用程序都在其自己的Android进程中运行,而这个进程只不过是一个Linux进程,而该进程首先需要一个执行线程。例如,当您单击电子邮件中的超链接时,网页将在浏览器窗口中打开。您的邮件客户端和浏览器是两个单独的应用程序,它们分别在两个单独的进程中运行。click事件使Android平台启动新进程,以便它可以在其自身进程的上下文中实例化浏览器活动。这对于应用程序中的任何其他组件同样适用。
Zygote:孕育新生命,新过程
让我们退后一会儿,快速浏览一下系统启动过程。与大多数基于Linux的系统一样,启动加载程序在启动时将加载内核并启动init进程。然后,init会生成称为“守护程序”的低级Linux进程,例如android debug守护程序,USB守护程序等。这些守护程序通常处理低级硬件接口,包括无线电接口。
然后,初始化过程会启动一个非常有趣的过程,称为“zygote'。
顾名思义,这是其余Android应用程序的开始。这是初始化Dalvik虚拟机的第一个实例的过程。它还预加载Android应用程序框架和系统上安装的各种应用程序使用的所有常见类。因此,它准备进行复制。它统计侦听套接字接口上的将来请求,以产生新的虚拟机(VM)来管理新的应用程序进程。收到新请求后,它会分叉以创建一个新进程,该进程将获取预先初始化的VM实例。
zygote之后,init启动运行时过程。
然后zygote分叉以启动一个名为System server的托管良好的进程。系统服务器在其自己的上下文中启动所有核心平台服务,例如活动管理器服务和硬件服务。
此时,完整的堆栈已准备就绪,可以启动第一个应用程序流程-主页应用程序,该应用程序显示主屏幕(也称为启动器应用程序)。
当用户单击启动器中的应用程序图标时…
click事件被转换为startActivity(intent),并通过Binder IPC路由到ActivityManagerService。ActvityManagerService执行多个步骤
- 第一步是收集有关Intent对象目标的信息。这是通过在PackageManager对象上使用resolveIntent()方法来完成的。默认情况下,使用PackageManager.MATCH_DEFAULT_ONLY和PackageManager.GET_SHARED_LIBRARY_FILES标志。
- 目标信息将保存回意图对象中,以避免重新执行此步骤。
- 下一个重要步骤是检查用户是否具有足够的特权来调用意图的目标组件。这是通过调用grantUriPermissionLocked()方法完成的。
- 如果用户具有足够的权限,ActivityManagerService会检查是否需要在新任务中启动目标活动。任务的创建取决于Intent标志(例如FLAG_ACTIVITY_NEW_TASK)和其他标志(例如FLAG_ACTIVITY_CLEAR_TOP)。
- 现在,它的时间来检查ProcessRecord已经存在process.If的ProcessRecord为空,则ActivityManager有可能创造一个新的进程实例化目标的组成部分。
如您所见,当用户单击图标并启动新应用程序时,许多事情发生在幕后。这是全图:
流程启动分为三个不同的阶段:
- 流程创建
- 装订申请
- 启动活动/启动服务/调用意图接收器…
流程创建:
ActivityManagerService通过调用startProcessLocked()方法创建一个新进程,该方法通过套接字连接将参数发送到Zygote进程。Zygote派生自己并调用ZygoteInit.main(),然后实例化ActivityThread对象并返回新创建的进程的进程ID。
默认情况下,每个进程都有一个线程。主线程有一个Looper实例来处理来自消息队列的消息,并且它在run()方法的每次迭代中都调用Looper.loop()。Looper的工作是从消息队列中弹出消息并调用相应的方法来处理它们。然后,ActivityThread通过随后调用Looper.prepareLoop()和Looper.loop()来启动消息循环。
以下序列详细捕获了调用序列:
<figcaption class="jv jw di dg dh jx jy bo b fc cp ga" data-selectable-paragraph="" style="box-sizing: inherit; font-weight: 400; font-family: sohne, "Helvetica Neue", Helvetica, Arial, sans-serif; line-height: 20px; margin-left: auto; margin-right: auto; max-width: 728px; font-size: 14px; color: rgb(117, 117, 117); margin-top: 10px; text-align: center;">Android应用启动:单击事件以执行Looper调用顺序</figcaption>
应用程序绑定:
下一步是将此新创建的过程附加到特定应用程序。这是通过在线程对象上调用bindApplication()来完成的。此方法将BIND_APPLICATION消息发送到消息队列。该消息由Handler对象检索,该对象随后调用handleMessage()方法以触发特定于消息的操作-handleBindApplication()。此方法调用makeApplication()方法,该方法将应用程序特定的类加载到内存中。
下图描述了该调用序列。
<figcaption class="jv jw di dg dh jx jy bo b fc cp ga" data-selectable-paragraph="" style="box-sizing: inherit; font-weight: 400; font-family: sohne, "Helvetica Neue", Helvetica, Arial, sans-serif; line-height: 20px; margin-left: auto; margin-right: auto; max-width: 728px; font-size: 14px; color: rgb(117, 117, 117); margin-top: 10px; text-align: center;">Android应用启动:BIND_APPLICATION消息处理</figcaption>
启动活动:
在上一步之后,系统包含负责应用程序的进程,并将应用程序类加载到进程的私有内存中。在新创建的流程和现有流程之间,启动活动的调用顺序很常见。
实际的启动过程从realStartActivity()方法开始,该方法在应用程序线程对象上调用sheduleLaunchActivity()。此方法将LAUNCH_ACTIVITY消息发送到消息队列。该消息由handleLaunchActivity()方法处理,如下所示。
假设用户单击“视频浏览器”应用程序。启动该活动的调用顺序如图所示。
<figcaption class="jv jw di dg dh jx jy bo b fc cp ga" data-selectable-paragraph="" style="box-sizing: inherit; font-weight: 400; font-family: sohne, "Helvetica Neue", Helvetica, Arial, sans-serif; line-height: 20px; margin-left: auto; margin-right: auto; max-width: 728px; font-size: 14px; color: rgb(117, 117, 117); margin-top: 10px; text-align: center;">Android应用启动:LAUNCH_ACTIVITY消息处理</figcaption>
该活动通过onCreate()方法调用开始其托管生命周期。该活动通过onRestart()调用进入前台,并通过onStart()调用开始与用户进行交互。