Android 应用点击图标到Activity界面显示的过程分析

分析这个过程不是单纯为走一遍函数调用的流程,而是更好的理解平常用到的一些方法、对象的初始化时间,对象创建的个数,方法的先后顺序,以及每个类,方法背后的作用和目的。主要是一下几个问题:

  1. Application是什么时候创建的,每个应用程序有几个Application
  2. 应用的资源路径什么时候初始化的
  3. 应用中ContextImpl的个数
  4. Application.attach(),Activity.attach()的调用时机及作用
  5. Instrumentation的重要性及具体作用
  6. 点击Launcher启动Activity和应用内部启动Activity的区别
1. 应用的安装过程

应用安装的时候,通过PMS解析apk的AndroidManifest.xml文件,提取出这个apk的信息写入到packages.xml文件中,这些信息包括:权限、应用包名、icon、APK的安装位置、版本、userID等等。packages.xml文件位于系统目录下/data/system/packages.xml。

2. 系统启动开启的服务

系统的会在启动时也可以认为开机时启动常用的服务,如ActivityManagerService(AMS),PackageManagerService(PMS),WindowManagerService(WMS),以及ServiceManager(SM),用于管理各种服务,详细的管理方式见理解Binder框架

同时桌面Launcher会为安装过的应用生成不同的应用入口,对应桌面上的应用图标,下面分析点击应用图标的到应用启动的过程。这里主要是应用端的过程,服务端也就是AMS少量涉及,同时以大体框架为主,不深入代码细节。主要分为Launcher进程,AMS进程,应用程序进程。

  • Instrumentation: 用于管理应用程序和系统(主要与应用程序内的Activity)的交互过程,Instrumentation将在任何应用程序运行前初始化,每个进程只会存在一个Instrumentation对象,且每个Activity都有此对象的引用,可以通过它监测系统与应用程序之间的所有交互,主要是内部交互。
  • **ActivityThread: **App的真正入口,通过调用main()App开始真正运行,同时开启消息循环队列,虽然不是一个真正的线程,但一般所在的线程被称为UI线程或主线程。ActivityThread就是专门与AMS的进行外部交互。
  • ApplicationThread: 应用需要和远程服务AMS等通信,而Binder只能单项通信,而AMS等服务想控制应用需要应用程序提供一个Binder接口,ApplicationThread就是这个Binder接口,用于通过远程服务调用本地的方法。
  • ActivityManagerProxy: AMS远程服务在本地的代理。
  • ApplicationThreadProxy: ApplicationThread在远程服务AMS的代理。

其实Launcher本身也是一个Activity,我们不考虑Launcher的创建过程,只分析用户的应用程序的从点击到Activity启动的过程。上面的这些每个进程只存在一个,如果应用存在多个进程,就会有多个实例。

Launcher所在进程,Launcher通过Binder与ActivityManagerService与通信。

  Launcher.startActivitySafely
  |
  Launcher.startActivity  
  |
  Activity.startActivity
  |   
  Activity.startActivityForResult
  | 
  Instrumentation.execStartActivity
  |
  ActivityManagerNative.getDefault().startActivity
  |  
  ActivityManagerProxy.startActivity

上面的这些都是在一条调用链上,ActivityManagerProxy是AMS的本地代理,实际的工作是在远程AMS完成的,下面是AMS进程。

  借助binder驱动
  ActivityManagerService.startActivity-> (AMS)  
  ...
  //一系类AMS的调用链和一些与Launcher通过Binder的互相调用过程,此时仍然未创建应用程序的进程。
  ...
  * AMS创建一个新的进程,用来启动一个ActivityThread实例, 
  * 即将要启动的Activity就是在这个ActivityThread实例中运行 
  Process.start("android.app.ActivityThread",...)->    
  // 通过zygote机制创建一个新的进程    
  Process.startViaZygote->调用新进程的main()
  ActivityThread.main->

创建新进程的时候,AMS会保存一个ProcessRecord信息,Activity应用程序中的AndroidManifest.xml配置文件中,我们没有指定Application标签的process属性,系统就会默认使用package的名称。每一个应用程序都有自己的uid,因此,这里uid + process的组合就可以为每一个应用程序创建一个ProcessRecord。每次在新建新进程前的时候会先判断这个ProcessRecord是否已存在,如果已经存在就不会新建进程了,这就属于应用内打开Activity的过程了。

AMS通过开启新的进程并调用ActivityThread.main后,接下来就是应用程序进程了。

public static void main(String[] args)  {
    Looper.prepareMainLooper();
    //又新建一个ActivityThread并调用attach(false)
    ActivityThread thread = new ActivityThread();
    thread.attach(false);
    if (sMainThreadHandler == null) {    
        sMainThreadHandler = thread.getHandler();
    }
}

thread.attach(false)主要是为了将ApplicationThread通过Binder驱动"传递"到远程AMS,也就是绑定,主要
是为了让AMS能通过ApplicationThread的代理ApplicationThreadProxy来调用ApplicationThread的方法,而本地应用程序通过ActivityManagerProxy来调用远程ActivityManagerService的方法,相当于应用程序与
AMS的通信窗口。

注意此时只创建了应用程序的ActivityThread和ApplicationThread,和开启了Handler消息循环机制,其他的都还未创建, ActivityThread.attach(false)又会最终到AMS的attachApplication,这个工程其实是将本地的ApplicationThread传递到AMS。然后AMS就可以通过ApplicationThread的代理ApplicationThreadProxy来调用应用程序ApplicationThread.bindApplication,通知应用程序的ApplicationThread已和AMS绑定,可以不借助其他进程帮助直接通信了。此时Launcher的任务也算是完成了。过程如下:

应用进程:
ActivityThread.attach
|    
IActivityManager.attachApplication(mAppThread)
| 
ActivityManagerProxy.attachApplication(mAppThread)   

AMS进程:
ActivityManagerService.attachApplication
|
ApplicationThreadProxy.bindApplication 

应用进程:

ApplicationThread.bindApplication
| Handler通信
AplicationThread.handlerBindApplication

ApplicationThreadProxy.bindApplication(...)会传来这个应用的一些信息,如ApplicationInfo,Configuration等,在ApplicationThread.bindApplication里会待信息封装成'AppBindData',通过

sendMessage(H.BIND_APPLICATION, data)

将信息放到应用里的消息队列里,通过Handler消息机制,在ActivityThread.handleMeaasge里处理H.BIND_APPLICATION的信息,调用AplicationThread.handleBindApplication。

handleBindApplication(AppBindData data) {
    Process.setArgV0(data.processName);//设置进程名
    ...
    //初始化mInstrumentation
    if(data.mInstrumentation!=null) {
        mInstrumentation = (Instrumentation)    cl.loadClass(data.instrumentationName.getClassName()).newInstance();
    }else {
        mInstrumentation = new Instrumentation();
    }

    //创建Application,data.info是个LoadedApk对象。
    Application app = data.info.makeApplication(data.restrictedBackupMode, null);
    mInitialApplication = app;

    //调用Application的onCreate()方法。
    mInstrumentation.callApplicationOnCreate(app);
}

LoadedApk类:
public Application makeApplication(boolean forceDefaultAppClass,Instrumentation instrumentation) {
    
    if (mApplication != null) {   
       return mApplication;
    }
    
    String appClass = mApplicationInfo.className;
    java.lang.ClassLoader cl = getClassLoader();
    
    //此时新建一个Application的ContextImpl对象,
    ContextImpl appContext = ContextImpl.createAppContext(mActivityThread, this);
    
    //通过在handleBindApplication创建的mInstrumentation对象新建一个Application对象,同时进行attach。
    app = mActivityThread.mInstrumentation.newApplication(cl, appClass, appContext);
    appContext.setOuterContext(app);
 }

Instrumentation类:
public Application newApplication(ClassLoader cl, String className, Context context) {    
    return newApplication(cl.loadClass(className), context);
}

Instrumentation类:
static public Application newApplication(Class<?> clazz, Context context)  {
    //实例化Application
    Application app = (Application)clazz.newInstance();     
    
    // Application和context绑定
    app.attach(context);    
    return app;
}

//attach就是将新建的ContextImpl赋值到mBase,这个ContextImpl对象就是所有Application内Context的具体
//实现,同时赋值一些其他的信息如mLoadedApk。
final void attach(Context context) {    
    mBase = base;  
    mLoadedApk = ContextImpl.getImpl(context).mPackageInfo;
}

这时Application就创建好了,这点很重要,很多博客里说Application是在performLaunchActivity里创建的,因为performLaunchActivity也有mInstrumentation.newApplication这个调用,newApplication函数中可看出会先判断是否以及创建了Application,如果之前已经创建,就返回已创建的Application对象。

ApplicationThreadProxy.bindApplication完成后,同时在AMS进程,调用realStartActivityLocked,接着就通过ApplicationThreadProxy调用到应用程序进程

//AMS进程:
ActivityManagerService.realStartActivityLocked
|
ApplicationThreadProxy.scheduleLaunchActivity

//应用程序进程
ApplicationThread.scheduleLaunchActivity
|  Handler通信
sendMessage(H.LAUNCH_ACTIVITY, r);
|  handleMassage
ActivityThread.handleLaunchActivity
|
ActivityThread.performLaunchActivity {
    //类似Application的创建过程,通过classLoader加载到activity.
    activity = mInstrumentation.newActivity(classLoader, 
               component.getClassName(), r.intent);
    //因为Activity有界面,所以其Context是ContextThemeWrapper类型,但实现类仍是ContextImpl.
    Context appContext = createBaseContextForActivity(r, activity);

    activity.attach(context,mInstrumentation,application,...);
    
    //attach后调用activity的activity方法。
    mInstrumentation.callActivityOnCreate(activity,...)
    
}

在ActivityThread.handleLaunchActivity里,接着调用
|
ActivityThread.performResumeActivity
|
activity.performResume
|
mInstrumentation.callActivityOnResume(this);
|
this.onResume;

activity.onResume就是和Window,View之类的绑定相关了,此时界面就显示出来了。
3. 总结
  1. Application是在ActivityThread.handleBindApplication中创建的,一个线程只会创建一个Application,但一个应用程序如果有多个进程将会创建多个Application对象。

  2. 应用资源是在Application初始化的时候,也就是创建Application ContextImpl的时候,ContextImpl就包含这个路径,主要就是对就是ResourcesManager这个单例的引用。

  3. 可以看出每次创建Application和Acitvity以及Service时就会有一个ContextImpl实例,ContentProvider和BroadcastReceiver的Context是其他地方传入的。所以Context数量=Application数量+Activity数量+Service数量,单进程情况下Application数量就是1。

  4. attach是依附、贴上的意思,可以理解为将两种事物联系在一起,上面分析的过程主要涉及3个attach,ActivityThread.attach可以理解为将应用程序进程的ApplicationThread依附到AMS,和AMS联系起来,用于整个AMS和应用程序的通信,Application.attach和Activity.attach主要是将新创建的ContextImpl对象与Application,Activity,Service等组件联系起来,对组件中其中的Context mBase变量赋值,以及一些初始化工作,比如MainHandler的赋值,mWindow的初始化(如果存在)等。ContextImpl包含资源信息、对Context的一些函数的实现等。
    可以很好的理解,attach需要在Application,Activity等组件调用onCreat之前调用,因为需要先完成组件的初始化工作。

  5. 管理着组件Application,Activity,Service等的创建,生命周期调用,很重要的一个类。例如:
    //创建Application
    mActivityThread.mInstrumentation.newApplication(cl, appClass, appContext);
    //调用Application的onCreate()方法。
    mInstrumentation.callApplicationOnCreate(app);

    //创建Activity,实际生命周期的管理。
    mInstrumentation.newActivity(classLoader, component.getClassName(), r.intent);
    mInstrumentation.callActivityOnCreate(activity);
    mInstrumentation.callActivityOnOnResume(activity);
    ...
    
  6. 点击Launcher时会创建一个新进程来开启Activity,而应用内打开Activity,如果Activity不指定新进程,将在原来进程打开,是否开启新进程实在AMS进行控制的,上面分析得到,每次开启新进程时会保存进程信息,默认为应用包名+应用UID,打开Activity时会检查请求方的信息来判断是否需要新开进程。Launcher打开Activity默认ACTIVITY_NEW_TASK,新开一个Activity栈来保存Activity的信息。

参考:【Android应用程序启动过程源代码分析

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容