美文网首页源码解析Android开发
Activity启动流程中关于LaunchMode为single

Activity启动流程中关于LaunchMode为single

作者: wodezhuanshu | 来源:发表于2016-03-17 17:04 被阅读310次

    Android应用程序启动流程源码分析(老罗)

    这篇文章主要是从源码角度去分析 为什么当我们设置了singleTask启动模式之后 activity 仍然在原来的activity的任务栈中添加新的activity 而不是新加一个

    首先我们调用startActivity(Intent,)在Activity的内部都会调用上面的方法

    我们看到首先Instrumentation的execStartActivity方法 我们主要看这个方法需要的参数

    this--->代表的是调用startActivity的Activity上下文信息

    mMainThread---->代表的是一个Binder对象 他是ActivityThread的一个成员变量  主要用于和 AMS进行进程间通信

    mTOken---->也是一个Binder的远程对象

    我们接下来来看Instrumentation的execStartActivity方法

    主要看这段代码:

    在这段代码里面  我们看到调用了ActivityManagerNative的getDefault方法 来获得一个AMS的远程代理对象

    我们看下这个函数的实现:

    通过单例模式 获得一个Activity在ServiceManager里面注册的activity服务对象 其实就是AMS  然后调用asInterface(b)来获得一个am的实例对象  其实就是

    获得了一个ActivityManagerProxy的对象 注意穿进去的是一个本地的Binder对象 就是通过这个对象发起进程间通信

    最后调用了AMP的startActivity进行进程间通信 最后还是要回调AMS的startActivity方法

    我们这里主要关注的是这个方法里的参数 至于Binder进程间通信如何进行 不在这里讨论

    whoThread-----就是IApplicationThread的binder对象 主要通过它于AMS进行进程间通信

    who-------就是上下文对象 其实就是Activity

    intent------就是目标Intent

    intent.resolveTypeIfneed-----返回的是null因为我们没有在Manifest文件里面注明Activity的mime类型

    token---- 就是一个binder的远程对象  其实就是MainActivity里面的一个Binder对象通过它可以获得MainActivity的相关信息 后边会保存到sourceRecord这个对象里面

    target-----就是我们发起调用的activity

    requestCode------如果不要求结果就是小于0

    0------flags

    ProfilerInfo ==null

    options 是一个Bundle的对象 这里也是null

    这些参数一一对象AMS的startActivity的参数

    这里我们主要关注几个参数就ok

    caller代表的就是ActivityThread里的成员变量IApplicationThread对象 他是一个binder对象

    intent 代表的就是我们要启动的activity的意图

    startFlags== 0

    requestCode = -1

    resultTo 是Activity的一个成员变量  是一个远程的binder对象(同上介绍)

    然后会调用

    这是ActivityStackSupervisor的一个方法

    我们来关注一下

    我们进入ActivityStackSupervisor

    首先会解析我们的Intent来获取信息

    ActivityInfo对象通过调用函数resoleIntent(args)方法

    这里的resultTo对象我们说过 这是MainActivity的一个成员变量  通过它我们能够获取到关于MainActivity的一些信息 

    这里两个ActivityRecord对象 有sourceRecord 和resultRecord

    sourceRecord代表的是启动目标activity的那个activity  这就是通过resultTO这个binder对象获得Mainactivity的相关信息然后保存到这个对象中

    resultRecord代表得是接受启动结果的Activity 因为requestcode==-1 所以这里resultRecord==null

    获取Intent的启动Flag 就是我们在Intent.setFlag里面设置的标志

    这个函数的主要作用就是处理sourceRecord和resultRecord两个对象

    在这里sourceRecord和resultRecord指向的应该是同一个activity

    因为当我们在MainActivity调用startActivity时候  我么的mainActivity是发起请求的Activity 而如果需要接受请求结果 那么也是我们的MainActivity

    所以他们两个的值一般是相同的。

    然后调用startActivityUncheckedLocked来处理本次的启动Activity的请求

    这个函数有点长 我们来分几部分来分析:

    首先获得我们的启动标志

    这里应为我们并没有设置这个FLAG_ACTIVITY_PREVIOUS_IS_TOP所以此时 我们的notTop==null

    startFlags == 0 所以此时不会进入这个判断

    这里我们将addingToTask标志置为了false默认是创建新的Task的

    然后进入else入口

    我们是在这里赋值的

    然后这里获取发起启动任务的Activity所在的Stack

    我们看下 为什么r.resultTO为null

    这个是在函数startActivityLocked函数中 构造ActivityRecord对象的

    这里因为我们不需要返回结构 所以这里requestcode肯定是小于0 的 所以resultRecord并没有被赋值 所以此时

    还是为null

    然后接下来的构造函数中

    我们红圈中的resultRecord就传给了我们的ActivityRecord  然后我们来看一下构造函数

    这里就很清楚了 然后我们接着往下分析

    因为我们设置的启动模式为singleTask 所以此时执行函数  findTaskLocked(r)

    这个函数会返回我们发起请求的MainActivity的实例

    这个函数中作用无非就是找到ID等于参数taskId的任务,然后在这个任务中查找是否已经存在即将要启动的Activity的实例,如果存在,就会把这个Actvity实例上面直到任务堆栈顶端的Activity通过调用finishActivityLocked函数将它们结束掉。在这个例子中,就是要在属性值affinity等于"xxx"的任务中看看是否存在SubActivity类型的实例,如果有,就把它上面的Activity都结束掉。这里,属性值affinity等于"xxx"的任务只有一个MainActivity,而且它不是SubActivity的实例,所以这个函数就返回null了。

    也就是找到这个任务栈存在的activity  然后清空他头顶的activity 让他自己在顶端 但是当我们初次打开时候 是没有的 所以此时应该返回null

    所以 此时将addintToTask=true  并且sourceRecord = 我们的MainActivity

    变量addingToTask值就为true了,同时将变量sourceRecord的值设置为taskTop,即前面调用findTaskLocked函数的返回值,这里,它就是表示MainActivity了。

    这里就是将targetStack赋值为发起请求的activity的任务栈 其实就是MainActivity所在的任务栈

    最后就走了我们正常的启动Activity的流程了

    这就是当我们设置activity的启动模式为singleTask时候的一些秘密所在

    即我们明明设置了singleTask 但是还是在原来activity的任务栈中添加新的activity

    相关文章

      网友评论

      • 孟买鱼:感谢楼主,分析的不错。这个方法太恶心了,看了好久越看越晕
        wodezhuanshu:@敬楠孟竟然 多看就好了 主要是了解流程 而不要过分关注细节

      本文标题:Activity启动流程中关于LaunchMode为single

      本文链接:https://www.haomeiwen.com/subject/pjtphttx.html