Android进阶学习第一天(三)Activity 启动模式

Activity的启动模式也是一个难点,主要原因是形形色色的启动模式和标志太容易被混淆了。

 Activity的LaunchMode

我们知道,默认情况下,当我们多次启动同一个Activity的时候,系统会创建多个实例并把他们一一放入任务栈中,当我们点击Back键时,会发现这些Activity会一一回退。任务栈是一种“后进先出”的栈结构。没按一次会有一个Activity出栈,知到栈空为止,当栈中午任何Activity的时候,系统就会自动回收这个任务栈。默认模式在多次启动同一个Activity时会创建多个实例,这样会显得很傻,当然为了避免这种情况,Android在设计的时候提供了启动模式来修改系统默认行为。目前有四种启动模式:standard、singleTop、singleTask、singleInstance。

 

1,standard:标准模式

也就是系统的默认模式,每启动一个Activity都会创建新的实例,不管实例是否已经存在。这是一种典型的多实例多实现,一个任务栈中可以有多个实例,每个实例也可以属于不用的任务栈。在这种模式下,谁启动这个Activity,那么这个Activity就运行在启动它的那个Activity所在的任务栈中,比如Activity A启动Activity B(B是标准模式) ,那么B就会进入A所在的栈中。如果当我们用ApplicationContext去启动standard模式的Activity的时候会报错。这是因为standard的Activity默认进入启动它的Activity所属的任务栈中,但是由于非Activity的Context并没有任务栈,所以会报错。

2,singleTop:栈顶复用模式

在这种模式下,如果新的Activity已经位于任务栈的栈顶,那么此Activity不会被重新创建,同时它的onNewIntent 方法会被回调,通过此方法的参数我们可以取出当前请求的信息。需要注意的是,这个Activity的onCreate和onStart不会被系统调用,因为它并没有发生变化,如果新的Activity的实例已经存在但是不是位于栈顶,那么新Activity还是会重建。比如,假设栈内是ABCD,其中ABCD为四个Activity,A在栈顶,D在栈顶,这时候再启动D,如果D启动模式为singleTop则栈内的情况还是ABCD。

3,singleTask:栈内复用模式

这是一种单实例模式,在这种模式下,只要Activity在一个栈中存在,那么多次启动这个Activity都不会重建,和singleTop一样,系统会调用onNewIntent。具体一点,当一个具有singleTask模式的Activity请求启动以后,比如ActivityA,系统会首先寻找是否存在ActivityA想存在的任务栈,,如果不存在,那就重新创建一个任务栈,然后创建A的实例后把A放入栈中。如果存在A所需要的任务栈,这时候要看A是否在任务栈中有实例存在 。如果存在,那么系统会吧A调到栈顶并调用它的onNewIntent方法,如果实例不存在,就创建A的实例并把A压入栈中。

举几个例子:

(1)比如现在任务栈S1中的情况为ABC,这时候Activity D以singleTask模式请求启动,其所需要的任务栈为S2,由于S2和D的实例均不存在,所以系统会先创建任务栈S2,然后创新D的实例并将其入栈到S2中。

(2)另一种情况,假设D所需的任务栈为S1,其他情况如1所示,因为S1已经存在了,所以系统会直接创建D的实例并将其入栈到S1中。

(3)如果D所需的任务栈为S1,并且当前S1栈内情况为ADBC,根据栈内复用的原则,此时D不会重新创建,系统会把D切换到栈顶,并调用其onNewIntent方法。同时singleTask默认具有clearTop的效果,会导致D上面的Activity全部出栈,于是最终S1栈内的情况为AD。

4,singleInstance:单实例模式

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

在singleTask启动模式中,多次提到了某个Activity所需的任务栈,其实也就是TaskAffinity,这个参数标识着一个Activity所需的任务栈的名称。默认情况下,所有Activity所需的任务栈名称都是应用的包名,当然我们也可以为Activity单独指定TaskAffinity属性,这个属性不能与包名相同,否则等于没有指定。TaskAffinity属性主要和singleTask启动模式或者allowTaskReparenting 属性配对使用,在其他情况下使用无意义。另外任务栈分为前台任务栈和后台任务栈。后台任务栈中的Activity位于暂停状态,用户可以通过切换将后台任务栈再次调到前台。

当TaskAffinity和singleTask启动模式配对使用的时候,他是具有该模式的Activity的目前任务栈的名字,待启动的Activity会运行在名字和TaskAffinity相同的任务栈中。

如何给Activity指定启动模式呢?有两种方法。

第一种在AndroidManifest中为Activity指定启动模式

     <activity
            android:launchMode="singleTask"
            android:name="com.demo.main.AddDeviceActivity" >

第二种是通过Intent中设置标签位来为Activity指定启动模式

    Intent intent=new Intent();
    intent.setClass(MainActivity.this,AddDeviceActivity.class);
    intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    startActivity(intent);

Activity中的Flags

 

1,FLAG_ACTIVITY_NEW_TASK

这个标记位的作用是为Activity指定“singleTask”启动模式,效果与在XML中指定该启动模式相同。

2,FLAG_ACTIVITY_SINGLE_TOP

这个标记位的作用是为Activity指定“singleTop”启动模式,效果与在XML中指定该启动模式相同。

3,FLAG_ACTIVITY_CLEAR_TOP

具有此标记的Activity启动时,处于同一任务栈中所有位于它之上的Activity都要出栈。这个模式一般需要和FLAG_ACTIVITY_NEW_TASK配合使用,在这种情况下,如果被启动的Activity已经存在,那么系统就会调用它的onNewIntent方法。如果采用standard模式启动,那么它连同它之上的Activity都要出栈,系统会创建新的Activity实例放入栈顶。

4,FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS

具有这个标记的Activity不会出现在历史Activity的列表中,不希望用户返回该Activity时可以用这个标记。等同于在XML中设置Activity的属性android:autoRemoveFromRecents="true"。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值