zoukankan      html  css  js  c++  java
  • Android中Activity启动模式详解

    Activity启动模式设置:
    
            <activity android:name=".MainActivity" android:launchMode="standard" />
    
    Activity的四种启动模式:
    
        1. standard
    
            默认启动模式,每次激活Activity时都会创建Activity,并放入任务栈中。
    
        2. singleTop
    
            如果在任务的栈顶正好存在该Activity的实例, 就重用该实例,否者就会创建新的实例并放入栈顶(即使栈中已经存在该Activity实例,只要不在栈顶,都会创建实例)。
    
        3. singleTask
    
            如果在栈中已经有该Activity的实例,就重用该实例(会调用实例的onNewIntent())。重用时,会让该实例回到栈顶,因此在它上面的实例将会被移除栈。如果栈中不存在该实例,将会创建新的实例放入栈中。 
    
        4. singleInstance
    
            在一个新栈中创建该Activity实例,并让多个应用共享改栈中的该Activity实例。一旦改模式的Activity的实例存在于某个栈中,任何应用再激活改Activity时都会重用该栈中的实例,其效果相当于多个应用程序共享一个应用,不管谁激活该Activity都会进入同一个应用中。


           大家遇到一个应用的Activity供多种方式调用启动的情况,多个调用希望只有一个Activity的实例存在,这就需要Activity的 onNewIntent(Intent intent)方法了。只要在Activity中加入自己的onNewIntent(intent)的实现加上Manifest中对Activity设置 lanuchMode=“singleTask”就可以。

           onNewIntent()非常好用,Activity第一启动的时候执行 onCreate()---->onStart()---->onResume()等后续生命周期函数,也就时说第一次启动Activity 并不会执行到onNewIntent(). 而后面如果再有想启动Activity的时候,那就是执行onNewIntent()---->onResart()------>onStart()----->onResume().  如果android系统由于内存不足把已存在Activity释放掉了,那么再次调用的时候会重新启动Activity即执行onCreate()---->onStart()---->onResume()等。

         当调用到onNewIntent(intent)的时候,需要在onNewIntent() 中使用setIntent(intent)赋值给Activity的Intent.否则,后续的getIntent()都是得到老的Intent。

    • FLAG_ACTIVITY_NEW_TASK

        在新的任务中启动activity-即不在本任务中启动.如果一个包含这个activity的任务已经在运行,那个任务就被弄到前台并恢复其UI状态,然后这个已存在的activityonNewIntent()中接收新的intent

        这个标志产生与"singleTask"相同的行为.

    • FLAG_ACTIVITY_SINGLE_TOP

        如果正启动的activity就是当前的activity(位于后退栈的顶端),那么这个已存在的实例就接收到onNewIntent()的调用,而不是创建一个新的实例.

        这产生与"singleTop"模式相同的行为.

    • FLAG_ACTIVITY_CLEAR_TOP

        如果要启动的activity已经在当前任务中运行,那么在它之上的所有其它的activity都被销毁掉,然后这个activity被恢复,而且通过onNewIntent()initent被发送到这个activity(现在位于顶部了)

        没有launchMode属性值对应这种行为.

        FLAG_ACTIVITY_CLEAR_TOP多数时候与FLAG_ACTIVITY_NEW_TASK联用.当一起使用时,会在其它任务中寻找一个已存在的activity实例并其把它放到一个可以响应intent的位置.

      注:如果Activity的启动模式是"standard"FLAG_ACTIVITY_CLEAR_TOP会导致已存在的activity被从栈中移除然后在这个位置创建一个新的实例来处理到来的intent.这是因为"standard"模式会导致总是为新的intent创建新的实例.

    Activity的两种启动模式:FLAG_ACTIVITY_CLEAR_TOP和FLAG_ACTIVITY_REORDER_TO_FRONT
     
    1. 如果已经启动了四个Activity:A,B,C和D。在D Activity里,我们要跳到B Activity,同时希望C finish掉,可以在startActivity(intent)里的intent里添加flags标记,如下所示:
    1 Intent intent = new Intent(this, B.class);
    2 intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
    3 startActivity(intent);

     

    这样启动B Activity,就会把D,C都finished掉,如果你的B Activity的启动模式是默认的(multiple) ,则B Activity会finished掉,再启动一个新的Activity B。
      如果不想重新再创建一个新的B Activity,则在上面的代码里再加上:


    1 intent.addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP);  


    这样B Activity就不会再创建一个新的了,而是会重用之前的B Activity,同时调用B Activity的onNewIntent()方法。
     


    2. 如果已经启动了四个Activity:A,B,C和D,在D Activity里,想再启动一个Actvity B,但不变成A,B,C,D,B,而是希望是A,C,D,B,则可以像下面写代码:


    1 Intent intent = new Intent(this, MainActivity.class);
    2 intent.addFlags(Intent.FLAG_ACTIVITY_REORDER_TO_FRONT);
    3 startActivity(intent);

    i.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    i.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);

    i.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK|Intent.FLAG_ACTIVITY_CLEAR_TOP);

    这两种方式 都不行,网上说可以,但我一直不行,不过FLAG_ACTIVITY_NEW_TASK在配置文件中设置,然后FLAG_ACTIVITY_CLEAR_TOP在flags中则可以

    有一种情况你会经常遇到,其它实体(如搜索管理器SearchManager 或者 通知管理器NotificationManager)会启动你的活动。这种情况下,你需要使用 Intent.FLAG_ACTIVITY_NEW_TASK 标记,因为活动在任务(这个应用/任务还没有被启动)之外被启动。就像之前描述的一样, 这种情况下标准特性就是当前和任务和新的活动的亲和性匹配的任务将会切换到前台,然后在最顶端启动一个新的活动。当然,你也可以实现其它类型的特性。

    一个常用的做法就是将Intent.FLAG_ACTIVITY_CLEAR_TOP 和NEW_TASK一起使用。这样做,如果你的任务已经处于运行中,任务将会被切换到前台来, 在栈里的所有的活动除了根活动,都将被清空,根活动的onNewIntent(Intent) 方法传入意图参数后被调用。当使用这种方法的时候 singleTop 或者 singleTask启动模式经常被使用,这样当前实例会被置入一个新的意图,而不是销毁原先的任务然后启动一个新的实例。

    另外你可以使用的一个方法是设置活动的任务亲和力为空字串(表示没有亲和力),然后设置finishOnBackground属性。 如果你想让用户给你提供一个单独的活动描述的通知,倒不如返回到应用的任务里,这个比较管用。要指定这个属性,不管用户使用BACK还是HOME,活动都会结束;如果这个属性没有指定,按HOME键将会导致活动以及任务还留在系统里,并且没有办法返回到该任务里。

    http://blog.csdn.net/niu_gao/article/details/7294949

    http://www.cnblogs.com/xiaoQLu/archive/2012/07/17/2595294.html

    http://blog.csdn.net/java2009cgh/article/details/7000821

    http://blog.csdn.net/findsafety/article/details/9664061

  • 相关阅读:
    多线程伪共享FalseSharing
    C语言restrict限定符
    Linux线程基础函数
    Linux信号函数
    C函数前向声明省略参数
    12.2 关闭DLM 自动收集统计信息 (SCM0)ORA-00600之[ksliwat: bad wait time]
    pdb的数量限制
    关闭或开启memory_target
    OSWATCH安装
    参数SID写错,ERROR OGG-00664 ORA-01034: ORACLE not available ORA-27101: shared memory realm does not exist
  • 原文地址:https://www.cnblogs.com/jiezzy/p/3704941.html
Copyright © 2011-2022 走看看