taskAffinity详解

60 篇文章 1 订阅
<activity
	android:name=".SecondActivity"
	android:launchMode="singleTask"
	android:taskAffinity="com.example.demo" />
  • 那taskAffinity含义:就是给Activity栈打一个tag。
  • 栈tag默认值是包名,也就是如果没有写这个属性,那默认的栈tag就是包名。

taskAffinity、launchMode的关系

  • 如果是android:launchMode="singleInstance",创建一个随机tag(可能是taskAffinity可能不是)单独栈。
  • 如果是launchMode不是singleInstance,先查找taskAffinity对应的栈tag是否已经存在,如果存在对应栈就使用该栈,如果不存在就创建一个tag为taskAffinity的栈,然后再根据launchMode模式对Activity进行栈操作。

备注:

  • singleInstance优先级最高,也就是如果同时指定android:launchMode="singleInstance"和指定taskAffinity,创建出来的栈的tag值不一定是taskAffinity的值。因为singleInstance是要单独栈的,如果存在了该taskAffinity的栈,就创建一个其他tag的栈,如果不存在则创建tag为taskAffinity的栈。也就是说这种情况,创建出来的栈的tag值可能是taskAffinity设置的值也可能不是。
  • 如果指定不是android:launchMode="singleInstance"模式,首先会根据taskAffinity查找对应tag的栈是否存在,如果存在就使用不存在就创建,然后根据android:launchMode进行Activity的栈操作(没有显示设置taskAffinity那该属性默认值就是包名)。

优先级:

      taskaffinity 可以单独对一个 activity 使用,代表该 activity 所想归属的 task;
也能对application 使用,代表该 application 内声明的所有 activity 都归属于这个task。

如果 activity 组件没有声明 taskAffinity 的话,该 activity 的 taskAffinity 属性也是有默认值的。

如果 application 指定了 taskAffinity 值,默认值就是 application 指定的 taskAffinity 值;

如果 application 未指定的话,默认值就是 manifest 中声明的包名(package 对应的字符串)。

是否开启新的Task:

是不是我指定了一个 Activity 的 taskAffinity 值(跟包名不同),运行该 Activity 时,是否就会新开这个 task栈呢?

答案是否定的,一个 Activity 运行时所归属的task,默认是启动它 的那个Activity 所在的 task(下文将会验证)。

taskAffinity生效

taskAffinity 单独使用并不会生效。
要想其生效,需要配合其他属性使用,比如:

配合 Intent.FLAG_ACTIVITY_NEW_TASK,或者配合 allowTaskReparenting 。或者配合android:launchMode="singleTask"

使用时用其中的一个就行,下面将详细介绍这两个属性。

1、

Intent intent = new Intent(this,BActivity.class);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(intent);

      当 AMS 发现启动了一个带有 FLAG_ACTIVITY_NEW_TASK 标签的 Activity 时,会先去寻找当前是否存在这个 Activity 的 task 值,如果不存在的话,就会创建该task,如果存在就省去了创建 task 这个步骤。然后在把要启动的 Activity 添加到 task 中。

2、

<activity android:name=".CActivity"
            android:taskAffinity="com.ddddd"
            android:launchMode="singleTask"
            android:label="CActivity">
</activity>

      会先去寻找当前是否存在这个 Activity 的 task 值,如果不存在的话,就会创建该task,如果存在就省去了创建 task 这个步骤(singleInstance无论android:taskAffinity="com.ddddd"任务栈是否存在,都会独立创建一个栈来存放当前activity;singleTop、standard都是默认是启动它 的那个Activity 所在的 task)

<activity
            android:label="${appName}"
            android:name=".MainActivity"
            android:taskAffinity="com.ddddd"
            android:exported="true">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <activity android:name=".BActivity"
            android:taskAffinity="com.ddd"
            android:launchMode="singleTask"
            android:label="BActivity"></activity>
        <activity android:name=".CActivity"
            android:taskAffinity="com.ddddd"
            android:launchMode="standard"
            android:label="CActivity"
             ></activity>
MainActivity->BActivity->CActivity:

 3、android:allowTaskReparenting="true"

<activity android:name=".SecondActivity"
            android:exported="true"
            android:allowTaskReparenting="true"></activity>

      allowTaskReparenting 这个属性指的是一个 Activity 运行时,可以重新选择自己所属的task。基本是在跨app 间调用时。比如:

                //隐士启动三方app,某页面
                Intent intent = new Intent();
                ComponentName componentName = new                 
                        ComponentName("com.example.demo1210",
                        "com.example.demo1210.SecondActivity");
                intent.setComponent(componentName);
                startActivity(intent);

用上面启动三方app后,如果再从桌面图标启动三方app,默认会到三方app的主Activity,如果三方app页面设置 android:allowTaskReparenting="true" ,再从桌面点击图标启动app,将会重新宿主到自己的任务栈中,也就会看到之前启动过的页面。

4、Android 任务栈简介

      一个Android应用程序功能通常会被拆分成多喝Activity,而各个Activity之间通过Intent进行连接,而Android系统,通过栈结构来保存整个App的Activity,栈低的元素是整个任务栈的发起者。一个合理的任务调度栈不仅是性能的保证,更是提供性能的基础。

      当一个App启动时,如果当前环境中不存在App的任务栈,那么系统就会创建一个任务栈。此后,这个App所启动的Activity都将在这个任务栈中被管理,这个栈也被称为一个Task,即表示若干个Activity的集合,他们组合在一起形成一个Task。另外,需要特别注意的是,一个Task中的Activity可以来自不同的App,同一个App的Activity也可能不在一个Task中。

     关于栈结构,相信大家都不会陌生——后进先出(Last In First Out)的线性表。根据Activity当前栈结构中的位置,来决定该Activity的状态。先来看看正常情况下的Android任务栈。当一个Activity启动了另一个Activity的时候,新启动的Activity就会置于任务栈的顶部端,并处于该活动,而启动它的Activity虽然功成身退,但依然保留在任务栈中,处于停止状态,当用户按下返回键或者调用finish()方法时,系统会移除顶部Activity,让后面的Activity恢复活动状态。当然世界不可能一直那么“和谐”,可以给Activity设置一些“特权”,来打破这种“和谐”的模式。这种特权,就是通过AndroidMainifest文件中的属性android:launchMode来设置或者是通过Intent的flag来设置

参考:

android开发android:taskAffinity标签属性的理解 - yongfengnice - 博客园

Activity启动源码阅读随笔_Alrey_的博客-CSDN博客

Android 任务栈简介_许诺承诺的博客-CSDN博客_android 任务栈

Android中的任务栈和Activity的启动模式

Android-taskAffinity

https://www.jianshu.com/p/bcb1db0677e7

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值