Android开发——Android系统启动以及APK安装、启动过程

摘要:
通常,在执行Bootloader程序后,将加载Android操作系统。它负责解析名为init.rc的配置文件,并启动在Android系统底部运行的一些服务进程。system_服务器将启动Android系统的本机服务和Java服务。到目前为止,从安卓手机开机到内核加载,最后到家庭桌面显示的整个过程已经结束。

0. 前言  

Android手机打开开关,到我们可以使用其中的app时,这个启动过程到底是怎么样的?

 

1.  系统上电

当给Android系统上电,在电源接通的瞬间,CPU内的寄存器和各引脚均会被置为初始状态,CPU复位之后,程序指针会指向启动地址,从该地址读取并直接运行启动程序的可执行代码,或者将可执行代码与数据载入CPU内置的RAM中再运行。这一段代码,放在PC中,叫做BIOS,而在Android等嵌入式系统中就叫做Bootloader

 

2.  Bootloader引导程序

通过Bootloader引导程序,可以初始化硬件设备,建立内存空间映射图,为操作系统内核准备好环境,完成整个系统的加载启动任务。

Android开发——Android系统启动以及APK安装、启动过程第1张

 Bootloader是支持交互式操作的,一般刷机中会用到的Fastboot模式,就是在Bootloader启动之后,根据用户输入的指令(在手机上用音量键,Home健等操作),停留在当前状态。

而正常情况下,Bootloader程序执行完成之后,就会开始加载Android操作系统

 

3.  加载内核

我们知道Android操作系统的内核其实就是Linux内核。当Bootlader引导程序启动之后,就会进入Linux的内核引导阶段。该阶段会初始化内核和设备驱动,接着启动内核,挂载文件系统,最后启动用户空间进程

 

4.  Init程序

Init程序是用户空间的第一个进程。它负责解析一个叫做init.rc的配置文件,启动Android系统底层运行的一些服务进程

 init.rc文件定义了不同的ActionService,其格式如下:

//Action是以关键字on来定义的,紧接着的boot是一个触发条件
//如下面的action表明当Android启动(boot)的时候,要做些什么事情
// Action boot之后的class_start core等,是Init程序在开机时会启动的服务
//这些服务包括ueventd, servicemanager, void, zygote,installd, ril-daemon等,我们一般称它们为守护进程
on boot 
    ... 
    class_start core 
    class_start main 
 
//Service是以关键字servie来定义的,紧跟着的是其名字和其启动的文件
//如下会执行/system/bin/serviemanager文件,并将其命名为servicemanager
service servicemanager/system/bin/servicemanager #声明一个service,而servicemanager则是其名字 
    class core                         #optioin, 该service属于core类 
     usersystem                         $option,该service属于System用户 
    group System 

Init程序同时也会开启属性服务,存放一些关于Android系统的全局系统设置信息。

最后,当所有的ActionService都启动完成之后,Init程序就会进入一个无限的循环,监听系统中发生的事件,对一些事件进行处理,比如重启某些服务等。

 

5.  ServiceManager

ServiceManagerInit程序启动的守护进程之一,它是AndroidBinder通信机制的基础

 ServiceManager提供了注册,检索服务的功能。在Zygote子进程system_server启动的每一项服务都会将其注册到ServiceManager中,由ServiceManager来统一进行管理。值得注意的是,ServiceManager本身也是一个Service,所以它也会将自己注册到自身上面。

 ServiceManager启动的过程做了三件事:

1)打开Binder设备,把Binder设备映射到内核空间中,由于内核空间是所有进程共享的区域,所以借助这块区域,可以实现不同进程间资源的共享,从而达到进程间通信。

2ServiceManager本身会变成一个ContextManager,即上面所说的注册到自己身上,负责各种Service的上下文管理

3)最终ServiceManager也是会进入一个无限循环

 

6.  Zygote

Zygote也是由Init程序启动的其中一个服务,是Android中的第一个Dalvik虚拟机。在之后,所有的Dalvik虚拟机都是由Zygote孵化出来的,这是因为Android中每一个应用都有着自己的进程,而每一个进程中都着一个Dalvik虚拟机,如果每起一个应用就创建一个虚拟机,效率太低,而通过复制Zygote,不仅可以提高创建的速度,还能够共享系统和框架的资源,可以大幅度提高应用程序启动的效率。

 Zygote启动之后:

1)首先会创建一个socket等待某个客户端的请求

2)接着创建其第一个子进程system_serversystem_server会开启Android系统的Native ServiceJava Service

其中Native Service就是那些利用JNI实现的共享库,而Java Service则是通过Java语言实现的ActivityManagerService等,都是在这个阶段由system_server启动并注册到一个叫ServiceManager的服务进程中。

ActivityManagerService启动之后,会进入到一个叫System Ready的状态,而在其达到System Ready的状态之前呢,就会启动一个叫做Launcher的应用,而这个Launcher就是Android系统启动成功之后见到的Home桌面了。

而在SystemReady的状态,其会跟Zygote启动之初的Socket进行通信,告诉Zygote一切已经就绪。

3Zygote知道已经启动成功,就开始进入一个无限的循环,在Socket上等待请求的到来。

 

至此,从Android手机上电的那一刻起,到加载内核,到最后显示Home桌面的整个过程就结束了。


7.  APK的安装过程

Android系统启动的过程中会启动一个应用程序管理服务PackageManagerService(系统启动的时候由SystemServer组件启动),这个服务负责扫描/system/framework、/system/app、/vendor/app、/data/app、/data/app-private五个目录下的APK文件,然后解析该APK中的AndroidManifest.xml以获得应用程序相关信息,其中最重要的就是依据sharedUserId这个配置来让系统确定每个APK运行在哪个进程。
继而为这个APK分配Linux用户ID、用户组ID以便APK在系统中可以获取到合适的运行权限、资源访问权限。
最后将之前获得的权限和APK安装信息保存到本地的一个配置文件中,以便下次在安装这些APK时可以将需要的APK信息很快速的恢复回来。


8.  应用启动流程

8.1  应用的冷热启动

Android中的冷启动指当启动应用时,后台没有该应用的进程,这时系统会重新创建一个新的进程分配给该应用所以会先创建和初始化Application类,再创建和初始化MainActivity类(包括一系列的测量、布局、绘制),最后显示在界面上。

Android中的热启动指当启动应用时,后台已有该应用的进程(如按back键、home键,应用虽然会退出,但是该应用的进程是依然会保留在后台,可进入任务列表查看),热启动就不会走Application这步了

 

8.2  应用的启动流程

当点击app图标时,系统会从Zygote进程中fork创建出一个新的进程分配给该应用,之后会依次创建和初始化Application类、创建MainActivity类、加载主题样式ThemeMainActivity以及配置Activity层级上的一些属性、再inflate布局、当onCreate/onStart/onResume方法都走完了后最后才进行contentViewmeasure/layout/draw显示在界面上,所以直到这里,应用的第一次启动才算完成,这时候我们看到的界面也就是所说的第一帧。

总结一下,应用的启动流程如下:

Application的构造器方法——>attachBaseContext()——>onCreate()——>Activity的构造方法——>onCreate()——>配置主题中背景等属性——>onStart()——>onResume()——>测量布局绘制显示在界面上


8.3  启动时间优化

可以通过adb shell命令的方式进行测量应用的启动时间:

adb shell am start -W [packageName]/[packageName.MainActivity]
应用初始化的时候采取以下策略可减少启动时间:

1)在Application的构造器方法、attachBaseContext()onCreate()方法中不要进行耗时操作的初始化。

2)对于MainActivity,由于在获取到第一帧前,需要对contentView进行测量布局绘制操作,尽量减少布局的层次,考虑StubView的延迟加载策略,当然onCreateonStartonResume方法中也要避免做耗时操作



感谢:

http://blog.csdn.net/linmiansheng/article/details/37728903

http://blog.csdn.net/u010687392/article/details/50518343

http://www.2cto.com/kf/201607/528367.html

http://www.jianshu.com/p/0b0d6f684580

Android开发——Android系统启动以及APK安装、启动过程第2张

免责声明:文章转载自《Android开发——Android系统启动以及APK安装、启动过程》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇【科创人独家】销售易CTO张忠:做产品不是翻译需求,用户思维绝非满足一切bootstrap-switch使用,small、mini设置下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

iOS性能优化:Instruments使用实战

Instruments使用技巧 关于Instruments官方有一个很有用的用户使用Guide,当然如果不习惯官方英文可以在这里找到中文本翻译版本PDF参阅.Instruments 确实是一个很强大的工具,用它来收集关于一个或多个系统进程的性能和行为的数据极为方便,并能及时跟踪随着时间产生的数据.还可以广泛收集不同类型的数据.关于Instrument工具基...

Android Native C 之 Helloworld的四种编译方式_转载

一.编写helloworld.c Android.mk     [root@fontlose jni]# cat hello.c  [cpp] view plaincopyprint?  #include <stdio.h>   int main()   {       printf("Hello World!\n");       ...

android 系统重启与关机:java 代码实现

在应用层如何通过代码实现 android 系统的重启,分享给大家。 这篇博客是在 android 系统开发的基础之上进行实践的,所以如果你是纯粹的 app 开发,可能要让你失望了。 该代码在真机上测试成功,在模拟器上面测试失败。 在 linux 下面,重启 pc,非 root 用户需要执行 sudo reboot,所以在 android 下重启机器也需...

Android中AsyncTask的简单用法

转自:http://blog.csdn.net/cjjky/article/details/6684959 在开发Android移动客户端的时候往往要使用多线程来进行操作,我们通常会将耗时的操作放在单独的线程执行,避免其占用主线程而给用户带来不好的用户体验。但是在子线程中无法去操作主线程(UI 线程),在子线程中操作UI线程会出现错误。因此android提...

在Tomcat服务器中启动SpringBoot项目原理(简化版)

总的来说,tomcat方式启动WAR包项目, tomcat会查询context上下文中实现ServletContainerInitializer接口的类,然后调用类的onStartup(Set<Class<?>> c, ServletContext ctx)方法 Spring的SpringServletContainerInitia...

Android ButterKnife注解式开发

在Android开发中findViewById和setOnClickListener解脱写法。 在任意的一个类中 @Bind(R.id.et) EditText editText; @OnClick(R.id.btn) public void btnClick(){ Toast.makeText(MainActivity.this,editText.get...