Activity启动模式singleTask的理解

Activity启动模式singleTask的理解1.Activity启动模式singleTask的理解2.Task与Activity栈3.singleTask设置方式4.singleTask的意义(作用)4.1.举例说明4.2.例子14.3.例子24.4.小结5.taskAffinity属性5.1.配置方式5.2.意义(作用)5.3.举例5.4.回退顺序5.5.总结1.Activity启动模式singleTask的理解2.Task与Activity栈Task是一些Acti..

大家好,又见面了,我是你们的朋友全栈君。

1. Activity启动模式singleTask的理解

2. Task与Activity栈

Task是一些Activity的集合,以Activity栈的形式存放。因此,Task是概念上的,Activity栈是实体上的。

可以说,新启动了一个Task就是新建了一个Activity栈,来存放这个Task内的Activity。

3. singleTask设置方式

  • manifest文件
    在activity节点里添加属性:
<activity
    android:name="com.example.test.SecondActivity"
    android:launchMode="singleTask"
/>

  • intent设置FLAG
Intent intent = new Intent(this, SecondActivity.class);
        intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
        startActivity(intent);

4. singleTask的意义(作用)

把一个activity启动模式设置为为singleTask,只是意味着framework在启动该activity时把它标识为可在一个新任务中启动,至于是否在一个新任务中启动,那可不一定了!是需要条件的,什么条件呢?请看下文分解~

4.1. 举例说明

4.2. 例子1

假设下面的跳转关系:

MainActivity -> SecondeActivity -> ThirdActivity

其中,SecondeActivity被设置了singleTask的启动模式,MainActivity和ThirdActivity都不做任何操作,保持默认,即standard模式

那么跳到ThirdActivity后,当前的Activity栈情况从顶到底部是:

ThirdActivity、SecondeActivity、MainActivity

也就是说依然按照正常压栈的顺序,和standard模式一样。并不是按照谷歌官方解释的,容易误解为新建了一个Task,并用新的栈存放SecondeActivity。这是误解。

那么,真的和standard模式一样吗?

还是有不同的,不然弄这个模式干嘛。

不同之处,用下面的例子说明:

4.3. 例子2

假设下面的跳转关系:

MainActivity -> SecondeActivity -> ThirdActivity -> ForthActivity ->SecondeActivity

其中,SecondeActivity被设置了singleTask的启动模式,其余Activity都不做任何操作,保持默认,即standard模式

那么跳到SecondeActivity后,当前的Activity栈情况从顶到底部是:

SecondeActivity、MainActivity

没错,SecondeActivity以上的 ThirdActivity、ForthActivity都被销毁了。

也就是说,重用了栈里面已有的 SecondeActivity ,并且把它上面的 activity 都清除了,从而使得它暴露在最上面。

顺便说一下,最后一个SecondeActivity的启动没有调用onCreate,而是依次调用了onRestart、onStart、onNewIntent、onResume这些生命周期方法。

4.4. 小结

被标记为singleTask的Activity,在启动的时候如果当前没有实例,就和standard模式一样,直接压栈到当前activity上;如果已经有实例了,就把该目标Activity实例上面的activity都清除,从而将目标Activity暴露到最上面,调用了其onNewIntent方法

5. taskAffinity属性

上面的两个例子都在我们认知范围内,还比较熟悉。但是这个taskAffinity想必就不太熟悉了。

taskAffinity是标记当前activity附属到哪个task上的。

5.1. 配置方式

  • manifest文件里
    在activty节点里配置该属性即可。taskAffinity取值是一个字符串,用来唯一标识一个Task的。

默认情况下,也就是不配置,taskAffinity取的是应用的包名。

 <activity android:name="com.example.test.SecondActivity"
             android:launchMode="singleTask"
             android:taskAffinity="com.example.test.second">
        </activity>

5.2. 意义(作用)

配合singleTask使用,意味着被这两个属性标记的activity可以被指定到一个新的task里。

5.3. 举例

假设下面的跳转关系:

MainActivity -> SecondeActivity -> ThirdActivity

其中,SecondeActivity被设置了singleTask的启动模式,并且配置了taskAffinity属性,MainActivity和ThirdActivity都不做任何操作,保持默认,即standard模式

那么跳到ThirdActivity后,当前的Activity栈情况从顶到底部是:

Task1: ThirdActivity、MainActivity

Task2: SecondeActivity

可以看到,SecondeActivity被创建到了新的栈里。

于是,可以得出之前疑问的答案,什么条件下设置为singleTask的activity会被创建到新的task呢?

答案是,同时被指定了另一个taskAffinity属性值的时候。

5.4. 回退顺序

现在处于onResume的activity是ThirdActivity,如果这时候按返回键呢?activity的显示顺序会是什么样呢?

答案正如上面Task1、Task2栈的排列顺序,ThirdActivity弹出后,MainActivity显示;再按返回键,MainActivity弹出后,SecondeActivity显示。

可以看出,新建的Task处于下方。

5.5. 总结

  1. 把启动模式设置为singleTask,framework在启动该activity时只会把它标示为可在一个新任务中启动,至于是否在一个新任务中启动,还要受其他条件的限制,这个条件就是taskAffinity属性配置为不同于包名的其他字符串。如果没有配置,默认就是当前包名,在当前Task内操作。
  2. 在启动一个singleTask的Activity实例时,如果系统中已经存在这样一个实例,就会将这个实例调度到任务栈的栈顶,并清除它当前所在任务中位于它上面的所有的activity。
    如果不存在,就看taskAffinity属性对应的Task是否存在,如果存在,就把Activity压入该Task内栈里;否则,就新建一个Task,再把Activity压入该Task内栈里。
  3. singeTask模式保证了在一个Task内只有一个activity实例

参考文章:https://blog.csdn.net/zhangjg_blog/article/details/10923643

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/153248.html原文链接:https://javaforall.net

(0)
上一篇 2022年6月26日 下午12:00
下一篇 2022年6月26日 下午12:00


相关推荐

  • 分布式文件存储的实现

    分布式文件存储的实现分布式文件存储的实现

    2022年5月9日
    44
  • springboot 事务配置

    springboot 事务配置1、全局配置@EnableTransactionManagement@Aspect@ConfigurationpublicclassGlobalTransactionConfig{//写事务的超时时间为10秒privatestaticfinalintTX_METHOD_TIMEOUT=10;//restful包下所有service包或者service的子包的任意类的任意方法privatestaticfinalStringAOP

    2022年5月17日
    57
  • C语言求最大公约数和最小公倍数(思路清晰+拓展)[通俗易懂]

    C语言求最大公约数和最小公倍数(思路清晰+拓展)[通俗易懂]最大公约数的求法首先了解它的一般求法(欧几里得算法):假设存在两个数A和B,假如A%B的结果不为0,那么A和B的最大公约数是B与A%B的最大公约数,一直往下计算,直到后者为0,此时的最大公约数为A’(注意不是A而是A’)。就比如上边的例子,当A%B==0的时候,最大公约数就是B了,这个A’就代表B。最大公约数的代码:(基于C++实现的函数)intgcd(inta,intb){ in…

    2022年5月16日
    54
  • 《VMware Virtual SAN权威指南(原书第2版)》一3.11 小结

    《VMware Virtual SAN权威指南(原书第2版)》一3.11 小结

    2022年3月5日
    58
  • 产品流量分析

    产品流量分析年底要接的数据需求好多,博客好久没更新了。这次和大家分享一下最近对流量分析的一些理解。流量是产品获得用户的第一步,没有流量就没有转化与营收。对于流量的分析在产品日常运营效果监控中有着非常重要意义。下面我们就流量的来源与流向分析中需要关注哪些指标,展开叙述。这里首先放一张对流量来源和去向的图:从流量来源角度来看,其来源包括直接访问、搜索访问、商务合作以及自媒体等方面:直接访问:用户直…

    2022年6月2日
    38
  • Oracle运算符

    Oracle运算符

    2021年8月15日
    50

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号