• 第一章04-Activity中常用的标志位


    Activity的LaunchMode

    Android中提供了四中Activity的启动模式

    1. standard
    2. singleTop
    3. singleTask
    4. signleInstance
    

    standard

    • 标准模式,这也是系统的默认模式,每次启动一个Activity都会创建一个新的实例,不管这个实例是否已经存在,Activity执行正常的生命周期方法
    • 一个任务栈中可以有多个实例,每个实例也可以属于不同的任务栈
    • 在这种模式下,谁启动了这个Activity,那么这个Activity就运行在启动它的那个Activity所在的栈中

    当我们使用ApplicationContext去启动standard模式的Activity的时候会报错

    报错原因:AndroidRuntimeException: Calling startActivity from outside of an Activity context requires the FLAG_ACTIVITY_NEW_TASK flag.

    • 这是因为standard模式的Activity默认会进入启动它的Activity所属的任务栈中,但是由于非Activity类型的Context(如ApplicationContext)并没有所谓的任务栈,所以这就有问题了
    • 解决这个问题的方法是为待启动Activity指定FLAG_ACTIVITY_NEW_TASK标志位,这样启动的时候就会为它创建一个新的任务栈
    • 这个时候待启动的Activity实际上是以singleTask模式启动的

    singleTop

    • 栈顶复用模式,在这种模式下,如果新的Activity已经位于任务栈的栈顶,那么此Activity不会被重新创建,同时它的onNewIntent方法会被回调,通过此方法的参数我们可以取出当前请求的信息
    • 需要注意的是,这个Activity的onCreate、onStart方法不会被系统调用,因为它并没有发生改变
    • 如果新Activity的实例已经存在但不是位于栈顶,那么新Activity仍然会重新创建

    singleTask

    • 栈内复用模式,这是一种单实例模式,在这种模式下,只要Activity在一个栈中存在,那么多次启动该Activity都不会重新创建实例,和singleTop一样,系统也会回调onNewIntent方法
    1. 当一个具有singleTask模式的Activity请求启动后,系统首先会寻找是否存在其想要的任务栈
    2. 如果不存在,就重新创建一个任务栈,然后创建A实例后把A放到栈中
    3. 如果存在,这是要看A是否在栈中有实例存在
      • 如果有A实例,那么系统就会把A调到栈顶并调用它的onNewIntent方法,由于singleTask默认具有clearTop的效果,会导致栈内所有在A上面的Activity全部出栈,这一点比较特殊
      • 如果不存在,就创建A的实例并把A压入栈中
    • 说明:
    • 什么是Activity所需要的任务栈呢?这要从一个参数说起,TaskAffinity,可以翻译为任务相关性
    • 这个参数标识了一个Activity所需要的任务栈的名字,默认情况下,所有Activity所需的任务栈的名字为应用的包名
    • 当然我们可以为每个Activity都单独指定TaskAffinity属性,这个属性值必须不能和包名相同,否则就相当于没有指定
    • TaskAffinity属性主要和singleTask启动模式和allowTaskReparenting属性配对使用,在其他情况下没有意义
    • 另外,任务栈分为前台任务栈和后台任务栈,后台任务栈的Activity位于暂停状态,用户可以通过切换将后台任务栈调到前台

    注意:taskAffinity属性的值为字符串,且中间必须含有包名分隔符"."

    singleTask模式的Activity切换到栈顶会导致在它之上的栈内Activity出栈,这是因为栈的工作模式是“后进先出”,栈内的元素要想回到栈顶,只能将它之前的Activity全部出栈,即CLEAR_TOP效果

    TaskAffinity和singleTask配对使用

    • 这个时候,它是具有该模式的Activity的目前任务栈的名字
    • 待启动的Activity会运行在名字和TaskAffinity相同的任务栈中

    TaskAffinity和allowTaskReparenting结合使用

    • 当一个应用A启动了应用B的某个Activity后,如果这个Activity的allowTaskReparenting属性为true的话,那么当应用B被启动后,此Activity会直接从应用A的任务栈转移到应用B的任务栈中
    • 可以这么理解:由于A启动了C,这时候C只能运行在A的任务栈中,但是C属于B应用,正常情况下它的taskAffinity值肯定不可能和A的任务栈相同,因为包名不同,所以当B启动后,B会创建自己的任务栈,这个时候系统发现C原本想要的任务栈已经被创建了,所以就把C从A的任务栈中转移过来了
    • 这种情况比较复杂,会产生特殊的效果

    singleInstance

    • 单实例模式,这是一种加强的singleTask模式,它除了具有singleTask模式的所有特性外,还加强了一点,那就是具有此模式的Activity,只能单独地位于一个任务栈中
    • 换句话说,比如A是singleInstance模式,当A启动后,系统会为它创建一个新的任务栈,然后A独自在这个新的任务栈中,由于栈内复用的特性,后续的请求均不会创建新的Activity,除非这个独特的任务栈被系统销毁了

    如何给Activity指定启动模式

    1. 通过AndroidMenifest为Activity指定启动模式
    2. 通过在Intent中设置标志位来为Activity指定启动模式

    这两种方式都可以为Activity指定启动模式,但是二者还是有区别的

    首先,优先级上,第二种方式的优先级要高于第一种,当两种同时存在时,以第二种方式为准

    其次,上述两种方式在限定范围上有所不同,比如,第一种方式无法直接为Activity设定FLAG_ACTIVITY_CLEAR_TOP标识,而第二种方式无法为Activity指定singleInstance模式

  • 相关阅读:
    Zookeeper入门(三)之工作流
    Zookeeper入门(二)之基础
    Zookeeper入门(一)之概述
    Docker删除/停止容器
    webbench安装和简单使用
    Notepad++ 7.3.2 Download 64-bit x64 / 32-bit x86
    7 常见问题
    6 完整测试
    5 安装Alloc服务
    4 安装MPush
  • 原文地址:https://www.cnblogs.com/coderwjq/p/6781505.html
Copyright © 2020-2023  润新知