startActivity简解

1.概要

startActivity启动一个activity的源码分析,涉及概念较多,比如binder,aidl跨进程通信(ipc),栈管理概念;启动的时序图如下图(来自其他blog)所示:

接下来基于sdk 23 源码里面看看,每个流程分别做了什么。主要分为APP进程部分Server进程部分

2.APP进程(上)

2.1 Context,ApplicationThread(在ActivityThread里面),Instrumentation

首先启动一个页面,比如activity里面通过context.startActivity,会调用到startActivityForResult方法,接着会调用Instrumentation的execStartActivity方法,里面有几个入参需要注意的,第一个context自身,一个是ApplicationThread(它是一个binder本地对象,或者说是server对象,它是抽象类ApplicationThreadNative的实现类,具体能力的承载类,为后续AMS持有它的proxy进行启动activity使用),还有一个参数intent(传递参数);

2.2 ActivityManagerNative,ActivityManagerService(AMS),ActivityManagerProxy(asInterface,它入参就是从sm那里拿来的binder代理对象),ServiceManager,

ServiceManager只是一个装饰类,最终调用的方法是binder内核驱动的方法,是通过BinderInterner里获取的binder对象,如:addService,registerService等方法。它对应的也有serviceManagerNative等,所有的binder服务(AMS/PMS/)等都是从这边获取到对应client对象。

通过ActivityManagerProxy.startActivity,他的里面调用transact,接着通过binder IPC进入AMS;到此为止都是在**APP进程中**

3.Server进程

--------------------Server进程部分-------------------

该部分主要是做一些启动前的校验,创建和管理Activity stack ,创建进程,绑定ApplicationThreadProxy.

3.1 AMN.onTransact

接下来我们就在, AMN是个抽象类,是个System server进程里面的本地服务,onTransact方法被调用,里面是真正的StartActivity方法的实现是在AMS中。

3.2 AMS.onTransact

在里面会调用startActivityAsUser,在SDK 23中是进入ActivityStackSupervisor,24中进入ActivityStarter的startActivityMayWait。

ActivityStackSupervisor为了多屏而出现,管理多个ActivityStack

3.3 ASS.startActivityMayWait

这个方法的作用主要是获取ActivityInfo信息(如果有多个activity会弹框让用户选择),里面会经过PMS获取ActivityInfo信息,并保存在intent对象中,接着进入startActivityLocked,它的返回值里,如果大于等于0标识启动成功,小于零标识启动失败。

3.4 ASS.startActivityLocked

该方法会进行一系列的常规检查,比如activity是否在AndroidManifest注册,是否存在这个activity等等,常规检查通过之后,会进行权限的检查;如果在都成功的情况下,进行ActivityRecrd(一个Activity可以对应多个ActivityRecord)的创建;并校验App切换是否允许,如果不允许则要启动的Activity添加到pending中,并直接返回,相当于检查下是否又被阻塞的activity,优先启动。

3.5 ASS.startActivityUncheckedLocked (栈管理相关的操作)

这个阶段就不需要做些权限的检查;这个函数的主要作用就是找到或者创建新的Activity所属于的task对象,之后调用ActivityStack的startActivityLocked。其中会根据我的ActivityRecord的属性,从标签中获取,进行Standard、singleTop、singleTask、singleInstance相关要求的处理。比如是否需要新建栈,或者已有栈中是否存在此Activity,如果应用之前没有起来过,那找不到task如何处理等等。

如果不存在该Activity,则需要创建进入

ActivityStack(栈的管理,activity是否显示等等)

3.6 AS.startActivityLocked

入参里面,ActivityRecord,是否需要立马显示,是否需要创建新的task等等。

如果新启动一个app,表示宿主栈ActivityStack中没有历史任务,或者强制要求在新的任务中启动Activity, 既然是一个新任务,那么就需要需要将任务插入宿主栈顶。放入并放在栈顶之后,将Activity推入显示状态。

3.7 AS.resumeTopActivityLocked-> resumeTopActivityInnerLocked

找到第一个没有finishing的栈顶activity,找不到直接回到桌面,找到了,就执行startPausingLocked暂停activity,接着进入startSpecificActivityLocked.

3.8 AS.resumeTopActivityInnerLocked

尝试将ActivityRecord迁移到显示状态(Resumed),如果发现待显示的Activity的宿主进程可能没有启动,则需要回到ASS.startSpecificActivityLocked。

ActivityStackSupervisor

3.9 ASS. startSpecificActivityLocked

如果通过包名,发现才宿主进程不存在,则启动一个新的应用进程。这时候会调用AMS.newProcessRecord来创建一个新的ProcessRecord,然后再调用另外一个重载的AMS.startProcessLocked()函数,来创建一个进程。进入AMS.startProccesslocked。如果已经存在则直接进入realStartActivityLocked方法,进行启动activity。

ActivityManagerService

3.10 AMS.startProccesslocked (参考:http://gityuan.com/2016/10/09/app-process-create-2/)

简单就是发送一个消息,通知Zygote进程fork一个应用进程,同时会调用入口程序的ActivityThread的main方法,该方法中会初始化几个比较的变量,如ApplicationThread,looper,H。以及非常重要的AT.attach过程,通过binder, 调用到AMP-> AMS.attachApplication, 其参数mAppThread的数据类型为ApplicationThread,将ApplicationThreadProxy绑定到AMS。

到此为止创建了待启动的Activity的宿主应用进程.

3.11 ASS.attachApplicationLocked

宿主进程如果创建好了,则调用realStartActivityLocked启动activity

4.APP进程(下)

--------------------回到APP进程部分-------------------

4.1 ApplicationThreadProxy.scheduleLaunchActivity

通过IPC就会回到app进程中,ApplicationThreadProxy.scheduleLaunchActivity回调ActivityThreadNative.onTransact,从而进入ApplicationThread的scheduleLaunchActivity进行真正启动Activity

4.2 ApplicationThread. scheduleLaunchActivity

通过H (handler)sendMessage(H.LAUNCH_ACTIVITY),进入handleLaunchActivity

4.3 ApplicationThread. handleLaunchActivity

通过performLaunchActivity获取一个Activity对象(反射等),具体如4.4,然后执行队形的oncreate,onstart, onresume等生命周期

4.4 ApplicationThread.performLaunchActivity

该函数负责实际执行应用进程中Activity的启动,是系统进程调度启动一个Activity的落脚地,也就是Activity生命周期的开始地方。一个新的Activity对象,是通过反射创建的,所以需要包名、类名等信息,部分信息已经从系统进程传递到应用进程了,部分信息也都可以通过PackageManager再向系统进程索取;新建一个Activity对象。有了参数以后,便可以通过ClassLoader加载到Activity对应的类,反射构建之。同时Activity的Context、Theme会在Activity对象构建之后被初始化;接着就开始Actiivty的生命周期调用如onCreate,onStart

4.5 ApplicationThread.handleResumeActivity -> performResumeActivity

直至Activity处于显示状态。

参考:http://gityuan.com/2016/03/12/start-activity/

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

推荐阅读更多精彩内容