activity的启动过程

简单的看acitivty的启动过程,就是从屏幕点击事件开始,也就是launcher组件发起,launcher组件通过AMS来启动的对应应用程序的MainActivity,准确的是清单文件中配置程序入口的那个activity,先暂时以MainActivity为例。

事件的发起

点击launcher的桌面图标时,它怎么知道该应用程序的相关信息呢?
答案是通过PackageManagerService,PMS在安装应用程序的时候会对应用程序的清单文件进行解析,从而得到它的组件信息。
接下来看launcher的源码

public final class Launcher extends Activity implement ...{
   
      void startActivitySafely(Intent intent,Object tag){
          intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK)
          try{
               startActivity(intent)
           } catch(ActivityNotFoundException e){
           ....
      }
}

首先Launcher是集成Activity,并且startActivitySafely方法最终也是调用了startActivity,而且intent的启动方式是新建栈,保证它可以在一个新的任务栈中启动

通知AMS

startActivity其实是一个父类的方法,间接调用了这个方法

mInstrumentation.ActivityResult ar = mInstrumentation.execStartActivity
(this,mMainThread.getApplication(),mToken,this,intent,requestCode)

这个方法主要关心两个类一个是mInstrumentation,一个是mToken,Instrumentation主要用来监控应用程序和系统之间的交互操作,mToken是IBinder对象,实际上这步就是Instrumentation通过进程间通信来通知AMS启动应用,实际上是先通知AMS的代理,然后又通过binder进程间通信通知AMS。

AMS通知ActivityThread

AMS内部做了什么事呢,它会先判断要启动的这个应用进程是否存在,如果不存在。会创建这个进程

public final class ActivityThread{

final AplicationThread mAppThread = new AplicationThread();

private final void attach(boolean system){
       ....
      mSystemThread = system;
     if(!system){
          IActivityManager  mgr =  ActivityManagerNative.getDefault();
          try{
                   mgr.attachApplication(mAppThread);
          } catch(RemoteException ex){
          }
     }
}

public static final void main(String[] args){    
  
      Looper.prepareMainLooper();

      ActivityThread  thread = new ActivityThread();
      thread.attach(false)
}

ActivityManagerNative通过getDefault()方法返回ActivityManagerService实例,ActivityManagerService通过attachApplication将ApplicationThread对象绑定到ActivityManagerService,而ApplicationThread作为Binder实现ActivityManagerService对应用进程的通信和控制。
创建进程时,会在进程中创建一个ActivityThread对象,并且调用它的成员函数attach向AMS发送一个启动完成的通知。
然后调用Looper的静态成员函数prepareMainLooper创建一个消息循环,并且在向AMS发送完成通知之后,使得当前进程进入到这个消息循环当中,ActivityThread中的AplicationThread本质是一个binder对象,AMS就是通过它和应用程序进程通信的。
,attach就是一个绑定方法,绑定成功之后通知ActivityStack启动Activity,接下来是一个调用链ActivityStack调用scheduleLaunchActivity通知ApplicationThread,ApplicationThread通过queueOrSendMessage通知ActivityThread,ActivityThread通过handleMessage收到launch消息,执行performLaunchActivity方法,该方法会调用activity的onCreate方法
重点看一下最后一步

ActivityThread启动activity

public final class ActivityThread{

       private final Activity performLaunchActivity(ActivityClientRecord r,Intent customIntent){
                Component component = r.intent.getComponent();
                Activity activity = null;
                try{
                     ClassLoader cl = r.package.getClassLoader();
                     activity = mInstrumentation.newActivty(cl,component.getClassName,r.intent);
               } catch (Exception ex){
               }
              try{
                    Application app  =  r.packageInfo.makeApplication(false,mInstrumentation);
                    if(activity != null){
                         ContextImpl appContext = new ContextImpl();
                         appContext.init(r.packageInfo,r.token,this);
                         appContext.setOuterContext(activity); 
                         activity.attach(appContext,this,getInstrumentation(),r.token,r.ident,app,r.intent,
                         r.activityInfo,title,r.parent,r.embeddedID,rlastNonConfigurationInstance,
                         r.lastNonConfigurationChildInstance,config)
                         mInstrumentation.callActivityOnCreate(activity,r.state);
                    }

               } catch (Exception ex){
               }
                return activity;
       }
}

ContextImpl创建的appContext用来作为actvity运行的上下文环境
mInstrumentation.callActivityOnCreate这个方法,最终由它调用了actvity的onCreate方法

参考自Android系统源代码情景分析,书中写的非常详细,而且包括launcher通知AMS,AMS收到通知去pause launcher,launcher收到消息执行,在通知AMS已经pause完成等等

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

推荐阅读更多精彩内容