Activity启动模式SingleTask和Intent.FLAG_ACTIVITY_CLEAR_TOP区别

Activity启动模式SingleTask和Intent.FLAG_ACTIVITY_CLEAR_TOP区别Activity启动模式介绍Android启动模式之前,先介绍两个概念task和taskAffinity task:翻译过来就是“任务”,是一组相互有关联的activity集合,可以理解为Activity是在task里面活动的。task存在于一个称为backstack的数据结构中,也就是说,task是以栈的形式去管理activity的,所以也叫可以称为“…

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

Activity启动模式

介绍 Android 启动模式之前,先介绍两个概念task和taskAffinity

  • task:翻译过来就是“任务”,是一组相互有关联的 activity 集合,可以理解为 Activity 是在 task 里面活动的。 task 存在于一个称为 back stack 的数据结构中,也就是说, task 是以栈的形式去管理 activity 的,所以也叫可以称为“任务栈”。

  • taskAffinity:官方文档解释是:”The task that the activity has an affinity for.”,可以翻译为 activity 相关或者亲和的任务,这个参数标识了一个 Activity 所需要的任务栈的名字。默认情况下,所有Activity所需的任务栈的名字为应用的包名。 taskAffinity 属性主要和 singleTask 启动模式或者 allowTaskReparenting 属性配对使用。

4种启动模式

  1. standard:标准模式,也是系统默认的启动模式,如果一个 Activity 的启动模式是 standard,则该 Activity 可以被多次实例化,且可以在不同的任务栈中存在。
    假如 activity A 启动了 activity B , activity B 则会运行在 activity A 所在的任务栈中。而且每次启动一个 Activity ,都会重新创建新的实例,不管这个实例在任务中是否已经存在。非 Activity 类型的 context (如 ApplicationContext )启动 standard 模式的 Activity 时会报错,因为非 Activity 类型的 context 并没有所谓的任务栈,所以系统会报错。此解决办法就是为待启动 Activity 指定 FLAG_ACTIVITY_NEW_TASK 标记位,这样启动的时候系统就会为它创建一个新的任务栈。这个时候待启动 Activity 其实是以 singleTask 模式启动的。

  2. singleTop:栈顶复用模式,如果一个 Activity 的启动模式是 singleTop,则该 Activity 可以被多次实例化,且可以在不同的任务栈中存在,且一个任务栈可以存在多个 singleTop启动模式的 activity。
    假如 activity A 启动了 activity B ,就会判断 A 所在的任务栈栈顶是否是 B 的实例。如果是,则不创建新的 activity B 实例而是直接引用这个栈顶实例,同时 B 的 onNewIntent 方法会被回调,通过该方法的参数可以取得当前 Intent 的信息;如果栈顶不是 activity B,则创建新的 activity B 实例压入栈(也就是一个任务栈存在多个实例)。

  3. singleTask:栈内复用模式。在第一次启动这个 Activity 时,系统便会创建一个新的任务栈,并且初始化 Activity 的实例,放在新任务栈的底部。但是,如果这个 Activity 已经存在于另一个任务栈中,则系统会销毁该 Activity 以上的所有Activity,然后调用该 Activity的 onNewIntent() 方法,不会创建新的实例。也就是说同一时刻只能存在一个该 Activity的实例。
    重要的一点就是 taskAffinity 属性。前面也说过了, taskAffinity 属性是和 singleTask 模式搭配使用的。

Activity启动模式SingleTask和Intent.FLAG_ACTIVITY_CLEAR_TOP区别

    4.singleInstance:单实例模式。这个是 singleTask 模式的加强版,它除了具有 singleTask 模式的所有特性外,它还有一点独特的特性,那就是此模式的 Activity 只能单独地位于一个任务栈,不与其他 Activity 共存于同一个任务栈。从这个 Activity 启动的任何其他Activity都会放到其他的任务栈。

还有一点:

无论 Activity 是在新任务栈中启动还是在相同的任务栈中启动,“返回”按钮始终会将用户带到上一个 Activity。但是,如果启动指定了 singleTask启动模式的 Activity,同时如果 后台 任务栈中存在该 Activity 的实例,则将整个任务栈内的 Activity 带到前台。此时,当前任务栈 现在包括在堆栈顶部提出的任务中的所有活动。如果不理解,下图说明这种情况。

Activity启动模式SingleTask和Intent.FLAG_ACTIVITY_CLEAR_TOP区别

 

通过上面我们了解了Activity的几种启动模式,接下来我们看看 SingleTask和 Intent.FLAG_ACTIVITY_CLEAR_TOP的区别:

singleTask:栈内复用模式。是一种单实例模式,在这种模式下,如果该Activity在栈中存在,那么多次启动此Activity都不会重新创建实例,而是销毁在它之上的所有Activity(不包括它本身),复用该Activity并调用它的onNewIntent方法,如果该Activity不存在,则创建该Activity并将其入栈到该Activity所需的任务栈中。

Intent.FLAG_ACTIVITY_CLEAR_TOP:销毁目标Activity和它之上的所有Activity,重新创建目标Activity,不会调用onNewIntent方法。

Intent.FLAG_ACTIVITY_SINGLE_TOP :当该activity处于栈顶时,可以复用,直接onNewIntent。

接下来我们看demo:

singleTask

在Manifest文件中我们将MainActivity指定为singleTask,

        <activity
            android:name=".MainActivity"
            android:configChanges="orientation"
            android:launchMode="singleTask">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <activity android:name=".SecondActivity" />
        <activity android:name=".ThirdActivity"></activity>

在每个Activity的onCreate方法中打印栈信息:

    private String getTaskInfo(){
        StringBuilder builder = new StringBuilder();
        ActivityManager am = (ActivityManager) getSystemService(Context.ACTIVITY_SERVICE);
        List<ActivityManager.RunningTaskInfo> taskInfos = am.getRunningTasks(10);
        for (ActivityManager.RunningTaskInfo task: taskInfos){
            builder.append("id= "+task.id+"\n");
            builder.append("description= "+task.description+"\n");
            builder.append("numActivityes= "+task.numActivities+"\n");
            builder.append("topActivity= "+task.topActivity+"\n");
            builder.append("baseActivity= "+task.baseActivity.toString()+"\n");
        }
        return builder.toString();
    }

MainActivity中设置一个Button启动SecondActivity:

Intent intent =new Intent(MainActivity.this,SecondActivity.class);
startActivity(intent);

SecondActivity中设置一个Button启动ThirdActivity:

Intent intent = new Intent(SecondActivity.this,ThirdActivity.class);
startActivity(intent);

SecondActivity中设置一个Button启动MainActivity,并传递消息:

Intent intent = new Intent(ThirdActivity.this,MainActivity.class);
startActivity(intent);

MainActivity的onNewIntent方法同样打印栈信息:

    protected void onNewIntent(Intent intent) {
        super.onNewIntent(intent);
        String taskInfo = getTaskInfo();
        Log.e(TAG, "onNewIntent: "+taskInfo);
    }

启动app,看打印出来什么东西

首先启动的是MainActivity,

MainActivity: onCreate: id= 1568
    description= null
    numActivityes= 1
    topActivity= ComponentInfo{com.example.hp.testtext/com.example.hp.testtext.MainActivity}
    baseActivity= ComponentInfo{com.example.hp.testtext/com.example.hp.testtext.MainActivity}
    id= 1559
    description= null
    numActivityes= 1
    topActivity= ComponentInfo{com.google.android.apps.nexuslauncher/com.google.android.apps.nexuslauncher.NexusLauncherActivity}
    baseActivity= ComponentInfo{com.google.android.apps.nexuslauncher/com.google.android.apps.nexuslauncher.NexusLauncherActivity}

可以看到MainActivity所在栈中Activity的数目numActivityes等于1,topActivity和baseActivity表示栈顶和栈底的Activity,

接下来启动SecondActivity,

SecondActivity: onCreate: id= 1568
    description= null
    numActivityes= 2
    topActivity= ComponentInfo{com.example.hp.testtext/com.example.hp.testtext.SecondActivity}
    baseActivity= ComponentInfo{com.example.hp.testtext/com.example.hp.testtext.MainActivity}
    id= 1559
    description= null
    numActivityes= 1
    topActivity= ComponentInfo{com.google.android.apps.nexuslauncher/com.google.android.apps.nexuslauncher.NexusLauncherActivity}
    baseActivity= ComponentInfo{com.google.android.apps.nexuslauncher/com.google.android.apps.nexuslauncher.NexusLauncherActivity}

可以看到栈中的Activity变为2

接下来启动ThirdActivity,

ThirdActivity: onCreate: id= 1568
    description= null
    numActivityes= 3
    topActivity= ComponentInfo{com.example.hp.testtext/com.example.hp.testtext.ThirdActivity}
    baseActivity= ComponentInfo{com.example.hp.testtext/com.example.hp.testtext.MainActivity}
    id= 1559
    description= null
    numActivityes= 1
    topActivity= ComponentInfo{com.google.android.apps.nexuslauncher/com.google.android.apps.nexuslauncher.NexusLauncherActivity}
    baseActivity= ComponentInfo{com.google.android.apps.nexuslauncher/com.google.android.apps.nexuslauncher.NexusLauncherActivity}

栈中的Activity变为3,

接着我们从ThirdActivity启动MainActivity,

MainActivity: onNewIntent: id= 1569
    description= null
    numActivityes= 1
    topActivity= ComponentInfo{com.example.hp.testtext/com.example.hp.testtext.MainActivity}
    baseActivity= ComponentInfo{com.example.hp.testtext/com.example.hp.testtext.MainActivity}
    id= 1559
    description= null
    numActivityes= 1
    topActivity= ComponentInfo{com.google.android.apps.nexuslauncher/com.google.android.apps.nexuslauncher.NexusLauncherActivity}
    baseActivity= ComponentInfo{com.google.android.apps.nexuslauncher/com.google.android.apps.nexuslauncher.NexusLauncherActivity}

我们发现这里只执行了onNewIntent方法,没有执行onCreate方法,说明服用了MainActivity,并且栈中只剩MainActivity,其他两个Activity都被销毁了。

Intent.FLAG_ACTIVITY_CLEAR_TOP

接下来我们把MainActivity的singleTask启动模式去掉(默认启动模式),并且ThirdActivity中启动MainActivity的代码如下:

Intent intent = new Intent(ThirdActivity.this,MainActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
startActivity(intent);

前面的步骤不变,当我们从ThirdActivity启动MainActivity时,打印的信息如下:

MainActivity: onCreate: id= 1570
    description= null
    numActivityes= 1
    topActivity= ComponentInfo{com.example.hp.testtext/com.example.hp.testtext.MainActivity}
    baseActivity= ComponentInfo{com.example.hp.testtext/com.example.hp.testtext.MainActivity}
    id= 1559
    description= null
    numActivityes= 1
    topActivity= ComponentInfo{com.google.android.apps.nexuslauncher/com.google.android.apps.nexuslauncher.NexusLauncherActivity}
    baseActivity= ComponentInfo{com.google.android.apps.nexuslauncher/com.google.android.apps.nexuslauncher.NexusLauncherActivity}

执行的是onCreate方法,说明MainActivity被重新创建了,同时其他两个Activity也是被销毁。

Intent.FLAG_ACTIVITY_CLEAR_TOP + Intent.FLAG_ACTIVITY_SINGLE_TOP

我们把ThirdActivity中的代码修改一下:

Intent intent = new Intent(ThirdActivity.this,MainActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP|Intent.FLAG_ACTIVITY_SINGLE_TOP);
startActivity(intent);

打印结果如下:

MainActivity: onNewIntent: id= 1571
    description= null
    numActivityes= 1
    topActivity= ComponentInfo{com.example.hp.testtext/com.example.hp.testtext.MainActivity}
    baseActivity= ComponentInfo{com.example.hp.testtext/com.example.hp.testtext.MainActivity}
    id= 1559
    description= null
    numActivityes= 1
    topActivity= ComponentInfo{com.google.android.apps.nexuslauncher/com.google.android.apps.nexuslauncher.NexusLauncherActivity}
    baseActivity= ComponentInfo{com.google.android.apps.nexuslauncher/com.google.android.apps.nexuslauncher.NexusLauncherActivity}

可以看到,使用Intent.FLAG_ACTIVITY_CLEAR_TOP + Intent.FLAG_ACTIVITY_SINGLE_TOP时调用了onNewIntent方法,复用了MainActivity,说明他们可以实现和singleTask一样的效果

总结

singleTask会销毁目标Activity之上的Activity并复用已经存在的目标Activity(调用onNewIntent),但Intent.FLAG_ACTIVITY_CLEAR_TOP会连同目标Activity一起销毁,然后重新创建目标Activity。

singleTask是写死在Manifest文件中的,如果觉得太笨重,可以同时使用Intent.FLAG_ACTIVITY_CLEAR_TOP + Intent.FLAG_ACTIVITY_SINGLE_TOP达到和singleTask一样的效果。

如果觉得不错顺手点个赞吧!!!

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

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

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • linux监控系统catic,网络设备监控-Catic添加H3C的监控图解[通俗易懂]

    linux监控系统catic,网络设备监控-Catic添加H3C的监控图解[通俗易懂]网络设备监控-Catic添加H3C的监控图解作者:尹正杰版权声明:原创作品,谢绝转载!否则将追究法律责任。首先,我要声明满足2个条件才能作本篇笔记的操作:第一:你得有台cacti服务器,第二:你得有台交换机,这台交换机需要配置团体名并且可以和你的cacti服务器互通。以上两点我已经在之前的笔记分享过,这里我就不啰嗦啦,直接上图。对了,如果你懒得部署Cacti的话,也可以直接用网上的别人做好的iso…

    2025年7月27日
    4
  • 查看数据库隔离级别,mysql

    查看数据库隔离级别,mysql1.查看当前会话隔离级别select@@tx_isolation;2.查看系统当前隔离级别select@@global.tx_isolation;3.设置当前会话隔离级别setsessiontransactionisolatinlevelrepeatableread;4.设置系统当前隔离级别setglobaltransac…

    2022年5月9日
    157
  • hibernate executeUpdate和executeNativeUpdate

    hibernate executeUpdate和executeNativeUpdateexecuteUpdate用来执行HQL的更新或者删除语句。executeNativeUpdate用来执行SQL的更新或者删除语句session.createQuery(hql)session.createSqlQuery(sql)最终都是通过ResultSetReturnImpl的executeUpdate(PreparedStatementstatement)方法实现publi…

    2022年10月20日
    5
  • Matlab中的数据预处理-归一化(mapminmax)与标准化(mapstd)

    Matlab中的数据预处理-归一化(mapminmax)与标准化(mapstd)最近遇到数据预处理的一些问题,本来很简单的东西,但是却搞的烦烦的,痛定思痛,决定自己实现一下。一、mapminmaxProcessmatricesbymappingrowminimumandmaximumvaluesto[-11]意思是将矩阵的每一行处理成[-1,1]区间,此时对于模式识别或者其他统计学来说,数据应该是每一列是一个样本,每一行是多个样本的同一维,即

    2022年6月15日
    34
  • 水流量霍尔传感器工作原理_51单片机温度传感器

    水流量霍尔传感器工作原理_51单片机温度传感器硬件准备(1)YF-B1流量传感器一个(2)51开发板一个基础知识(1)YF-B1流量传感器只有三根线。即数据线、VCC、GND。数据线输出为占空比为50%的方波。当水流通过水流转子组件时,磁性转子转动并且转速随着流量的变化而成线性变化。霍尔传感器(霍尔元件采样)输出相应的脉冲信号。其流量脉冲特性计算公式为:脉冲f(Hz)=8.1x流量Q(L/min)-3。(2)51单片机的中断…

    2022年9月29日
    4
  • foc无刷电机位置控制(直流无刷电机接线图解)

    序:矢量控制的核心思想是为了简化无刷电机的控制模型,将一个需要换相的无刷电机通过各种算法变换,抽象为一个直流电机的控制模型,只需要控制简单的两个直流分量来控制无刷电机,其中Vq抽象为直流电机的两端电压,Vd可调节电机力矩,但这个模型需要一个实时的电机轴角度θ参与计算。为了实现这个直流电机的控制模型,需要用到两个数学变换,即clarke变换和park变换。需要用到最原始的PID控制器等内容。…

    2022年4月11日
    385

发表回复

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

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