1. 需求背景
默认的启动activity是每次都会创建新的实例并把他们放入到任务栈中,然后,当你回退的时候,你会发现,一层又一层
2.解决方案:提供了可供选择的启动模式
- standard:标准模式,即是默认模式,新开一个就会执行创建操作,这种模式下,新启动的activity会加入到启动它的activity的栈顶。这里可能也会有个面试题,比如,用applicationcontext 启动activity可以吗?不可以,以为applicationcontext 没有任务栈,报错信息会提示需要 FLAG_ACTIVITY_NEW_TASK flag。当你为activity指定 这个flag时,系统就会为activity创建一个新的任务栈
Intent intent = new Intent(getApplicationContext(),B.class);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(intent);
- singleTask:栈内复用模式,很明显,就是同一个栈内activity只允许存在一个实例。指定了singleTask的activity启动过程(粗略描述,详解见下一节)描述如下:
- 系统首先寻找是否存在activity 想要的任务栈,如果不存在,创建新的任务栈,然后创建activity,压入到栈内。
- 如果存在所需任务栈
2.1.任务栈内是否已有activity的实例,如果没有,创建新的activity实例并压入到栈。
2.2. 任务栈内已有activity的实例,会把activity调到任务栈的栈顶,如果activity已经存在于栈内,那么会调用一个可能你不知道的方法onNewIntent
,在这里你可以获取到新的intent数据。同时将此activity之上(栈结构)的activity全部出栈(销毁)。
- activity想要的任务栈,这个算题外话,在manifest文件的activity标签下,有一个属性叫 taskAffinity(任务亲和),这四个字我理解为对某个任务(任务栈) 的"亲和度",亲和度高,那么activity就会在这个任务栈中启动。默认情况下,这个东西的值就是我们的程序包名。具有相同taskAffinity的activity会运行在相同的任务中(我感觉就像是进程一样,指定了相同名字的service运行在相同的进程里),这样会有一个有意思的现象,A应用里有activity A1,B里有B1(mainactivity),B2(子activity,allowTaskReparenting = true),A1,B2 的 taskAffinity相同。先用A1启动B2,然后回退到桌面,打开B应用之后,会展示B2,并不会展示B1,因为B2所需的任务栈已经创建了.另外还有一种,A 里有A1,A2,B里有B1,B2,A2,B2 的taskAffinity相同,且B2 allowTaskReparenting = true,这样启动A1,A2,B1,B2,启动顺序没问题,当你回退的时候,B2->A2->B1,奇怪吧,自己试验哦,这就说明了 任务这个概念,能跨应用,也能跨进程。感觉对一般人来说然并卵,用不到。ok,回归正题。
- singleTop:栈顶复用模式,顾名思义,只有在栈顶的时候才会被复用,其余时候都不会,就是如果栈内有activity,但是并不处于栈顶,还会创建个新的实力出来的。
*singleInstance:单实例模式,这种模式启动的activity会单独创建一个新的任务栈和一个新的实例(别误会,只在第一次创建的时候),应用场景书上说是为了和其他应用程序共享activity实例。
3 Activity 的启动方式
感觉并不够一篇文章,正好把启动方式拉过来就差不多了。
- 显示调用
启动一个activity主要分为两种方式,显示调用和隐式调用,其中显示调用简单明了,需要明确指定被启动的activity的类名,如下, intent 显示启动应该是最简单,最熟悉的了
Intent intent = new Intent(MainActivity.this,Main2Activity.class);
startActivity(intent);
这是最简单的,但是有一点需要注意,当内部类中调用时,第一个参数需要带上当前activity的类名。
- 隐式调用
隐式调用的好处就是能够解耦合,你只需要按照别人给出的文档进行数据组合,就能启动对应的activity,甚至你都不知道人家activity的类名。
IntentFilter,这个东西主要是用来过滤匹配你的intent,通过它能够使你的intent找到对应的activity然后启动,他有三个东西
- action:配置在manifest文件里至少有一个相同,多个相同会弹出对话框让你选择。注意区分大小写
注意,隐式启动的category 必须设置default,不然不会生效, 启动activity除外
why?
因为系统在调用startActivity
时,会默认的为我们加上这个,看代码, 而category的匹配规则又要求必须匹配,不写当然就会报错了
-
category:配置在manifest里,你得intent可以不设置category(使用系统默认的),但是如果你有了,不管几条,每一条都必须能在配置文件里的有相同的。
就是intent里的category 是manifest里的category的真子集。同样区分大小写。- data 这个东西的匹配原则跟activity一样,至少能匹配上一个就可以启动,只是data的结构复杂一点。
<data
android:scheme="string"
android:host="string"
android:port="string"
android:path="string"
android:pathPattern="string"
android:pathPrefix="string"
android:mimeType="string"
/>
就是这个里面有这么多东西需要设置,但是不用全设置上,我这个顺序是按照实际顺序排列的,实际intent中的data数据格式如下,其中scheme和port 缺少一个就会导致配置的data失效
scheme://host:port/path or pathPrefix or pathPattern
其中mimeType
并不在前边,因为这个东西代表的是媒体类型(必须是正式的,不能自己瞎写)。给出一个例子如下
Intent intent = new Intent("com.test.cn.intent.main2");
intent.setDataAndType(Uri.parse(""),"image/*");
startActivity(intent);
配置文件如下
<activity android:name=".Main3Activity">
<intent-filter>
<action android:name="com.test.cn.intent.main2" />
<category android:name="android.intent.category.DEFAULT" />
<data android:mimeType="image/*"/>
</intent-filter>
</activity>
这样就会启动Main3activity,下面在给出一个uri(Universal Resource Identifier 通用资源标识符,定位资源,现在明白为什么会有mimetype了吧,标出资源的类型)的例子。
Intent intent = new Intent("com.test.cn.intent.main2");
intent.setDataAndType(Uri.parse("123://"),"image/*");
startActivity(intent);
<intent-filter>
<action android:name="com.test.cn.intent.main2" />
<category android:name="android.intent.category.DEFAULT" />
<data android:mimeType="image/*"/>
<data android:scheme="123"/>
</intent-filter>
scheme可以瞎写,但是要注意的是,string里的scheme在转化之前要带://
这个东西,不然找不到
host 也可以瞎写,剩下的就更随意了,只要注意格式即可,格式如下
scheme(如果有后面的酒必须有它) ://host(如果有后面的酒必须有它) : port/ path(完整路径)或者pathpattern(通配符,需要注意正则表达式)或者pathPrefix(路径的前缀信息)
后两种我个人基本没用过。差不多就是这个样子了。
-------------------------------未完待续---------------------------------------------