从App启动到Launcher页面启动的流程

应用进程启动流程

先放一张应用进程启动的流程图。

当 Launcher应用点击某个应用图标时,会发送消息给 AMS,启动这个应用的 Launcher Activity。假如此时这个应用还未启动,AMS 会给 Zygote进程发送请求,要求先创建进程。 Zygote进程 会 fork出 一个进程,同时会对进程做一些初始化工作。之后这个会对这个新创建的进程会 启动binder机制,之后会执行应用进程入口函数:ActivityThread#main(进程会有一个默认线程创建去执行本进程的代码,这个默认的进程称之为主进程)在应用程序入口函数会启动主线程消息队列的Looper循环,之后会报告 AMS,进程已经启动完成可以进行接下来工作了。AMS 收到消息后,给应用进程发送消息,应用进程收到消息会干两件事:1、创建Application实例。2、启动 Launcher Activity。

以上是整个启动流程,具体从 ActivityThread#main 入口之后干得事情,就以代码片段的形式贴出来,来捋一下上述过程。

基于安卓版本:26

ActivityThread#main

public static void main(String[] args) {
  Looper.prepareMainLooper();
  ActivityThread thread = new ActivityThread();
  thread.attach(false);
  Looper.loop();
  ...
}

ActivityThread#attach

private void attach(boolean system) {
  final IActivityManager mgr = ActivityManager.getService();
  mgr.attachApplication(mAppThread);    // 进行IPC通信,通知AMS
  ...
}

ActivityManagerService#attachApplicationLocked

private final boolean attachApplicationLocked(IApplicationThread thread,int pid) {
  ...
  thread.bindApplication(...);    // 1、回调ActivityThread
  ... 
  mStackSupervisor.attachApplicationLocked(app)  // 2、启动首页
}

attachApplicationLocked 这里会有两个步骤需要注意的:1、回调App进程去创建Application 2、启动Launcher Activity

先看第一步:回调App进程去创建Application

ApplicationThread#bindApplication

public final void bindApplication(...){
  ...
  sendMessage(H.BIND_APPLICATION, data);
}

因为通过Binder机制进行IPC调用时,方法都在Binder线程中运行。通过Handler机制,将消息发送给主线程去执行。之后会调用 handleBindApplication方法

ActivityThread#handleBindApplication

private void handleBindApplication(...){
  ...
  //创建Application对象,并回调attachBaseContext
  Application app = data.info.makeApplication(data.restrictedBackupMode, null);  
  //回调 onCreate
   mInstrumentation.callApplicationOnCreate(app);
}

LoadedApk#makeApplication

public Application makeApplication(...){
  ...
  // 创建上下文 Context
  ContextImpl appContext = ContextImpl.createAppContext(mActivityThread, this);
  // 创建Application。并调用 attach、attachBaseContext来设置 mBase
  app = mActivityThread.mInstrumentation.newApplication(
                    cl, appClass, appContext);
}

至此,Application创建完成,同时通过 attachBaseContext 给mBase 设置了Application上下文Context:ContextImpl,之后调用了 onCreate声明周期方法。

再第二步:启动Launcher Activity

ActivityStackSupervisor#attachApplicationLocked

boolean attachApplicationLocked(...){
  ...
  ActivityRecord hr = stack.topRunningActivityLocked();
  realStartActivityLocked(hr, app, true, true)
}

ActivityStackSupervisor#realStartActivityLocked

final boolean realStartActivityLocked(...){
  ...
  // IPC 调用app进程ApplicationThread 启动页面
  app.thread.scheduleLaunchActivity(new Intent(r.intent), r.appToken,...);
}

Application#scheduleLaunchActivity

public final void scheduleLaunchActivity(...){
  ...
  sendMessage(H.LAUNCH_ACTIVITY, r);
}

接收到 AMS 启动页面的消息后,通过Handler机制把消息发送给主线程去执行:handleLaunchActivity

ActivityThread#handleLaunchActivity

private void handleLaunchActivity(...) {
  ...
  // 创建Activity,并执行attach、onCreate方法回调
  Activity a = performLaunchActivity(r, customIntent);
  // 执行 onResume 方法回调
  handleResumeActivity(r.token, ...);
}

ActivityThread#performLaunchActivity

private Activity performLaunchActivity(...){
  ...
  // 创建Activity 对应上下文环境 Context
  ContextImpl appContext = createBaseContextForActivity(r);
  // 创建Activity
  activity = mInstrumentation.newActivity(
                    cl, component.getClassName(), r.intent);
  // 调用attach方法,初始化工作,其中就有设置 mBase
  activity.attach(appContext, this, getInstrumentation(), r.token,
                        r.ident, app, r.intent, r.activityInfo, title, r.parent,
                        r.embeddedID, r.lastNonConfigurationInstances, config,
                        r.referrer, r.voiceInteractor, window, r.configCallback);
  // 回调 onCreate方法
  mInstrumentation.callActivityOnCreate(activity, r.state);
}

ActivityThread#handleResumeActivity

private Activity handleResumeActivity(...){
  ...
  r = performResumeActivity(token, clearHide, reason);
}
public final ActivityClientRecord performResumeActivity(...){
  ...
  r.activity.performResume();  // 调用onResume 回调
}

至此,启动的页面 onCreate 、 onResume 都已执行完成。

之前会在进程创建时,启动Binder机制,具体怎么启动呢,配一张图:


Binder机制启动

整体的主进程启动流程如下:


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