来也匆匆 几十繁华一阵风; 去也匆匆 回头一看已成空
分类: Android平台
2015-10-09 23:03:06
本文摘自:http://www.cnblogs.com/lwbqqyumidi/p/3775479.html
Task是一个具有栈结构的容器,可以放置多个Activity实例。启动一个应用,系统就会为之创建一个task,来放置根Activity;默认情况下,一个Activity启动另一个Activity时,两个Activity是放置在同一个task中的,后者被压入前者所在的task栈,当用户按下后退键,后者从task被弹出,前者又显示在幕前,特别是启动其他应用中的Activity时,两个Activity对用户来说就好像是属于同一个应用;系统task和task之间是互相独立的,当我们运行一个应用时,按下Home键回到主屏,启动另一个应用,这个过程中,之前的task被转移到后台,新的task被转移到前台,其根Activity也会显示到幕前,过了一会之后,在此按下Home键回到主屏,再选择之前的应用,之前的task会被转移到前台,系统仍然保留着task内的所有Activity实例,而那个新的task会被转移到后台,如果这时用户再做后退等动作,就是针对该task内部进行操作了。
我们今天就讲一下和task相关的知识,主要分一下几点:
1.Activity的affinity(亲和力)
2.Intent几种常见的flags
3.
affinity:
拥有相同affinity的多个Activity理论同属于一个task,task自身的affinity决定于根Activity的affinity值。affinity在什么场合应用呢?
1.根据affinity重新为Activity选择宿主task(与allowTaskReparenting属性配合工作);
2.启动一个Activity过程中Intent使用了FLAG_ACTIVITY_NEW_TASK标记,根据affinity查找或创建一个新的具有对应affinity的task。我们会在后面进行详细讲解。
默认情况下,一个应用内的所有Activity都具有相同的affinity,都是从Application(参考
1 <application android:name="Browser" 2 android:label="@string/application_name" 3 android:icon="@drawable/ic_launcher_browser" 4 android:backupAgent=".BrowserBackupAgent" 5 android:taskAffinity="android.task.browser" >
Intent几种常见的flags:
在android.content.Intent中定义了若干个flags,其中最重要的有以下几个:
1.FLAG_ACTIVITY_NEW_TASK:
当Intent对象包含这个标记时,系统会寻找或创建一个新的task来放置目标Activity,寻找时依据目标Activity的taskAffinity属性进行匹配,如果找到一个task的taskAffinity与之相同,就将目标Activity压入此task中,如果查找无果,则创建一个新的task,并将该task的taskAffinity设置为目标Activity的taskActivity,将目标Activity放置于此task。注意,如果同一个应用中Activity的taskAffinity都使用默认值或都设置相同值时,应用内的Activity之间的跳转使用这个标记是没有意义的,因为当前应用task就是目标Activity最好的宿主。
下面我们会通过实例进行演示这个特性:
我们新建两个项目,分别命名为appA和appB,并且分别创建FirstActivity和SecondActivity,我们准备让appB中的FirstActivity跳转到appA的SecondActivity。appA中的SecondActivity配置如下:
1 <activity android:name=".SecondActivity"> 2 <intent-filter> 3 <action android:name="android.intent.action.APP_A_SECOND_ACTIVITY" /> 4 <category android:name="android.intent.category.DEFAULT" /> 5 intent-filter> 6 activity>
然后,在appB中的FirstActivity跳转代码如下:
1 Intent intent = new Intent("android.intent.action.APP_A_SECOND_ACTIVITY"); 2 startActivity(intent);
我们要演示几个步骤:1.在appB中的FirstActivity点击按钮跳转到appA中的SecondActivity;2.按Home键回到主屏,在主选单中再次启动appB;3.按Home键回到主屏,在主选单中启动appA。演示过程如图所示:
再次启动appB应用:
启动appA应用:
我们发现在从appB跳转到appA的SecondActivity之后,SecondActivity实例好像是嵌入到了appB中,但是不影响appA的正常运行,这种关系如下图所示:
然后我们修改一下跳转的代码:
1 Intent intent = new Intent("android.intent.action.APP_A_SECOND_ACTIVITY"); 2 intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK); 3 startActivity(intent);
我们加上了FLAG_NEW_TASK标记,在来看一下演示结果:
再次启动appB:
启动appA:
我们看到差别了吧,当我们再次启动appB时已经看不到刚才启动的appA中的SecondActivity,而启动appA时却直接看到了,说明这个SecondActivity实例并不在appB的task内,而是创建了一个task,这个task的affinity就是SecondActivity默认的affinity,由于appA的SecondActivity的affinity是从Application继承而来,所以当appA启动时会直接找到这个task,而不是创建新的task。我们看一下解析图:
由此可见, FLAG_ACTIVITY_NEW_TASK应该这样去理解:根据Activity Affinity判断是否需要创建新的Task,然后再创建新的Activit实例放进去。