- Android进阶解密
- 刘望舒
- 1835字
- 2020-08-27 17:04:55
2.4 Launcher启动过程
此前已经学习了Android系统启动流程的init进程、Zygote进程和SystemServer进程,最后我们来学习Launcher的启动过程。
2.4.1 Launcher概述
系统启动的最后一步是启动一个应用程序用来显示系统中已经安装的应用程序,这个应用程序就叫作Launcher。Launcher在启动过程中会请求PackageManagerService返回系统中已经安装的应用程序的信息,并将这些信息封装成一个快捷图标列表显示在系统屏幕上,这样用户可以通过点击这些快捷图标来启动相应的应用程序。
通俗来讲Launcher就是Android系统的桌面,它的作用主要有以下两点:
(1)作为Android系统的启动器,用于启动应用程序。
(2)作为Android系统的桌面,用于显示和管理应用程序的快捷图标或者其他桌面组件。
2.4.2 Launcher启动过程介绍
SystemServer进程在启动的过程中会启动PackageManagerService,PackageManagerService启动后会将系统中的应用程序安装完成。在此前已经启动的AMS会将Launcher启动起来。Launcher启动过程的时序图如图2-3所示。
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer760.jpg?sign=1739345501-o38KWdRmdiiwb9EJu21sJCNv5d5wQU0p-0-3a5ea42932d9071c89a05223ab87e12f)
图2-3 Launcher启动过程时序图
启动Launcher的入口为AMS的systemReady方法,它在SystemServer的startOtherServices方法中被调用,如下所示:
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer761.jpg?sign=1739345501-Dm3S3PRb1NESc4W9AOPUZsbywgtMleIF-0-33770a8e0ad19326d2e2ddf8bd54fd93)
与Android 7.1.2源码不同的是,Android 8.0的部分源码引入了Java Lambda表达式,比如注释1处。下面来查看AMS的systemReady方法做了什么:
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer762.jpg?sign=1739345501-iLCSh6mvtaqwX7g7sxmWR6gmadXZGWNX-0-4e1d14c7ff80815602fd1b3126e8fa87)
systemReady方法中调用了ActivityStackSupervisor的resumeFocusedStackTopActivityLocked方法:
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer763.jpg?sign=1739345501-PW9Lo7xA533EtR3MDgxF7HpYHlopw3Ss-0-edd9d0c456e31b96c49bf8d9b6b6a805)
在注释1处调用ActivityStack的resumeTopActivityUncheckedLocked方法,ActivityStack对象是用来描述Activity堆栈的,resumeTopActivityUncheckedLocked方法如下所示:
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer764.jpg?sign=1739345501-UFF9Im3rXSWuMOkT2hQrZZ2q7ge5uB9W-0-45e05c79baa6993535e8cfd6896bc435)
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer765.jpg?sign=1739345501-hrJwqNeBFMxUEFjp0Mpg2bN6KYmhKmjU-0-a702e6e105bc77bbf488140a238e9bc5)
在注释1处调用了resumeTopActivityInnerLocked方法,如下所示:
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer766.jpg?sign=1739345501-4InRyTiVAtiWY0BaIeL3EqqfgI6XScXd-0-8ec9ac6d6ee9a004c7e7cb113c282426)
resumeTopActivityInnerLocked方法的代码很长,在此仅截取我们要分析的关键的部分,调用ActivityStackSupervisor的resumeHomeStackTask方法,代码如下所示:
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer767.jpg?sign=1739345501-jC8iM1ZKCoBrWKeFwlDvKBrhVfD0mbvE-0-729372baa212dc1c49acf0169fbe1cc1)
在resumeHomeStackTask方法中调用了AMS的startHomeActivityLocked方法,如下所示:
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer768.jpg?sign=1739345501-Z9v8ndPNL5jDsiJutpaFJGnSJIS1EI8Y-0-8adefc7a317180a0629777f6dea32c05)
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer769.jpg?sign=1739345501-HbxwuPy0JarVOTz0RHUi11r7rfdMxhho-0-31f14c4d706d7014664f411d9e32f20b)
注释1处的mFactoryTest代表系统的运行模式,系统的运行模式分为三种,分别是非工厂模式、低级工厂模式和高级工厂模式,mTopAction 则用来描述第一个被启动Activity组件的Action,它的默认值为Intent.ACTION_MAIN。因此注释1处的代码的意思就是mFactoryTest 为FactoryTest.FACTORY_TEST_LOW_LEVEL (低级工厂模式)并且mTopAction等于null时,直接返回false。注释2处的getHomeIntent方法如下所示:
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer770.jpg?sign=1739345501-mjyxJoZh96abllju2r4RVANV0X4WVnEr-0-b10a34156c718108ea2e35a0b3f44403)
在getHomeIntent方法中创建了Intent,并将mTopAction和mTopData传入。mTopAction的值为Intent.ACTION_MAIN,并且如果系统运行模式不是低级工厂模式,则将intent的Category 设置为Intent.CATEGORY_HOME,最后返回该Intent,再回到AMS的startHomeActivityLocked 方法,假设系统的运行模式不是低级工厂模式,在注释3处判断符合Action为Intent.ACTION_MAIN、Category为Intent.CATEGORY_HOME的应用程序是否已经启动,如果没启动则调用注释4处的方法启动该应用程序。这个被启动的应用程序就是Launcher,因为Launcher的AndroidManifest文件中的intent-filter标签匹配了Action为Intent.ACTION_MAIN,Category为Intent.CATEGORY_HOME。
Launcher的AndroidManifest文件如下所示:
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer771.jpg?sign=1739345501-djXZND1l8siCmUHzfN7UlmMidvWyQCR7-0-b6cd80f4e8fbe0b710af38717de072a5)
可以看到intent-filter设置了android.intent.category.HOME属性,这样名称为com.android.launcher3.Launcher的Activity 就成为了主Activity。从前面AMS的startHomeActivityLocked 方法的注释4处,我们得知如果Launcher没有启动就会调用ActivityStarter的startHomeActivityLocked方法来启动Launcher,如下所示:
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer772.jpg?sign=1739345501-VctdWYhGKypifUvw0ZqVJhOOC8dCeKCZ-0-290891f738dd65438ed115f8e03fad74)
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer773.jpg?sign=1739345501-k7ykn3YZPE98XN9P1fgH7S7K10L5Sazv-0-a70db22cd4a9772a742f251837d38ca0)
在注释1处将Launcher放入HomeStack中,HomeStack是在ActivityStackSupervisor中定义的用于存储Launcher的变量。接着调用startActivityLocked方法来启动Launcher,剩余的过程会和Activity的启动过程类似,本书将在第4章介绍该知识点。最终进入Launcher的onCreate方法中,到这里Launcher就完成了启动。
2.4.3 Launcher中应用图标显示过程
Launcher 完成启动后会做很多的工作,作为桌面它会显示应用程序图标,这与应用程序开发有所关联,应用程序图标是用户进入应用程序的入口,因此我们有必要了解Launcher是如何显示应用程序图标的。
我们先从Launcher的onCreate方法入手,代码如下所示:
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer774.jpg?sign=1739345501-TACv5K8eqUgA9gXbPdC660c3BxTaluCj-0-abde9b4e46cc7695c30924da79e216f0)
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer775.jpg?sign=1739345501-sOfX9taITnM8X1iprG998Zj1IRDec5Mh-0-27b09f35994690c2bdd7bf923546aadd)
在注释1处获取LauncherAppState的实例,在注释2处调用它的setLauncher方法并将Launcher对象传入,LauncherAppState的setLauncher方法如下所示:
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer776.jpg?sign=1739345501-97HqZOq0OsUYs32L1TC7dNbXRKFp0uQn-0-2f31b6537be09fa1ed5da261d882d612)
在注释1处会调用LauncherModel的initialize方法:
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer777.jpg?sign=1739345501-0XzvSCLN3AdKhi6OJxlyOuwIRXhjwydz-0-0643ffcb3b8bae16cb5056eb4191861e)
在initialize方法中会将Callbacks,也就是传入的Launcher,封装成一个弱引用对象。因此我们得知mCallbacks变量指的就是封装成弱引用对象的Launcher,这个mCallbacks③后面会用到它。
再回到Launcher的onCreate方法,在注释3处调用了LauncherModel的startLoader方法,如下所示:
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer778.jpg?sign=1739345501-94zze8QYyAS3Ti7D6EelSVz2skNBNiJu-0-60042028419385e9e08cbbfebeba76e7)
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer779.jpg?sign=1739345501-hTonBPD5NGu37myvDOcahPmT5BuZPG3X-0-360d32c5a72918b27dd6772f3bd3862b)
在注释1处创建了具有消息循环的线程HandlerThread对象。在注释2处创建了Handler,并且传入HandlerThread的Looper,这里Hander的作用就是向HandlerThread发送消息。在注释3处创建LoaderTask,在注释4处将LoaderTask作为消息发送给HandlerThread。
LoaderTask类实现了Runnable接口,当LoaderTask所描述的消息被处理时,则会调用它的run方法,LoaderTask是LauncherModel的内部类,代码如下所示:
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer780.jpg?sign=1739345501-qlExoY11GZiX3QT6m0YR8OnweEQTI29N-0-ef6770365bbddb93ee36654a54f11c4f)
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer781.jpg?sign=1739345501-oprYz8qSwOXDMhEWN181NTmsKv1Qli0Z-0-695c18bb17156c94d27810376e058c1f)
Launcher 是用工作区的形式来显示系统安装的应用程序的快捷图标的,每一个工作区都是用来描述一个抽象桌面的,它由n个屏幕组成,每个屏幕又分为n个单元格,每个单元格用来显示一个应用程序的快捷图标。在注释1处和注释2处分别调用loadWorkspace方法和bindWorkspace方法来加载和绑定工作区信息。注释3处的loadAllApps方法用来加载系统已经安装的应用程序信息,代码如下所示:
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer782.jpg?sign=1739345501-BPUzWjT368PFUr9ODFysAabRNgQm4B9E-0-0214110b20b7d8dd030d732a67432efd)
在注释1处会调用callbacks的bindAllApplications 方法,从之前的标注③处我们得知这个callbacks实际是指向Launcher的,下面我们来查看Launcher的bindAllApplications方法:
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer783.jpg?sign=1739345501-CCCj0dtXUWhtvN21sfEy5ZI4l9tifnM1-0-98ee839fe106f95f3aeb88db663f1b12)
在注释1处会调用AllAppsContainerView类型的mAppsView对象的setApps方法,并将包含应用信息的列表apps传进去,AllAppsContainerView的setApps方法如下所示:
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer784.jpg?sign=1739345501-YU2Wp5wGDzf6InTG5RNvoGtW1t0lGVoo-0-bb8781b66b87a847e097e2388460bd1a)
setApps 方法会将包含应用信息的列表apps设置给mApps,这个mApps是AlphabeticalAppsList类型的对象。接着查看AllAppsContainerView的onFinishInflate方法如下所示:
![](https://epubservercos.yuewen.com/D63A94/16896237205618706/epubprivate/OEBPS/Images/figer785.jpg?sign=1739345501-SdXI0lsJOsdcKInHGlUoW77zj9Xz438X-0-77fc52b4355b58b8fccfb3c2e58939e7)
onFinishInflate方法会在AllAppsContainerView加载完XML布局时调用,在注释1处得到AllAppsRecyclerView用来显示App列表,并在注释2处将此前的mApps设置进去,在注释3处为AllAppsRecyclerView设置Adapter。这样显示应用程序快捷图标的列表就会显示在屏幕上。
到这里Launcher中应用图标显示过程以及Launcher启动流程就讲解完了,接下来介绍Android系统启动流程。