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

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

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

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

发表回复

您的电子邮箱地址不会被公开。

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