四种启动模式
- standard
- singleTop
- singleTask
- singleInstance
第一个没什么好说的
第二个 如果Activity在栈顶,就不会创建新的 应用场景 读取通知页面
第三个 singleTask
可以理解为栈内的单例模式,在一个栈内只允许存在一个这个Activity的实例,singleTask 启动的时候 如果设置了taskAffinity 就先找taskAffinity的栈存在不,如果不存在就创建,如果存在就看栈里有没有实例,如果有就放到栈顶,并把原来该Activity上面的全pop掉,执行onNewIntent。
说到这该说一下 taskAffinity 了。
taskAffinity 用来指定Activity栈的名称,一般情况下,只有启动模式为singltTask和singleInstance的时候才起作用,但是当 启动Activity的intent 添加 FLAG_ACTIVITY_NEW_TASK flag的时候,也会起作用,下面可以验证下
<activity
android:name=".LauncherActivity"
android:taskAffinity="test.test1"
android:exported="true">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<activity
android:name=".MainActivity"
android:taskAffinity="test.test"
android:launchMode="singleTask">
<intent-filter>
<action android:name="android.intent.action.VIEW" />
<category android:name="android.intent.category.DEFAULT" />
<category android:name="android.intent.category.BROWSABLE" />
<data android:scheme="test" />
</intent-filter>
</activity>
<activity android:name=".activity.FragmentTestActivity"
android:taskAffinity="test.test1"
></activity>
LauncherActivity 加上 android:taskAffinity="test.test1"
MainActivity android:taskAffinity="test.test"
FragmentActivity android:taskAffinity="test.test1"
在Launcher跳转到MainActivity 然后再跳转到FragmentActivity 后,使用 adb shell dumpsys activity 看一下返回栈情况
image.png image.png可以看到 只有LauncherActivity 在 test.test1中,其他两个在test.test中。
为啥这样呢,LauncherActivity的启动模式为standard啊,而且FragmentActivity不是指定了taskAffinity了吗?
这就验证了刚才的结论了,LauncherActivity作为 启动页,系统Launcher启动它的时候会加上FLAG_ACTIVITY_NEW_TASK flag,所以taskAffinity 对它起作用,而FragmentActivity 就是个普通的Activity,所以不起作用,自然和启动它的Activity放进同一个返回栈了。
另外由于MainActivity是 singleTask的,所以它被放在了指定的taskAffinity中了。
第四个singleInstance
指的是整个系统中的单例,它不仅表示自己在一个栈里是单例的,而且它这个栈只能包含它一个Activity。
我们把上面的MainActivity改成singleInstance,然后删掉FragmentTestActivity的 taskAffinity
image.png
发现FragmentTestActivity 被放在了默认栈里,所以singleInstance启动的Activity会放在新的栈里。
如果给FragmentTestActivity 加上taskAffinity 并且和MainActivity的一样
image.png
image.png
发现FragmentTestActivity和MainActivity 的afinity相同,但是栈的id不同,这两个还是不在一个栈里。
走到这就迷糊了,既然taskAfinity都一样,那么栈咋不一样呢,taskAffinity和栈有什么关系?
这一点我真搞不明白了,taskAfinity不就是为了让Activity分组吗,相同的taskAffinity咋不同组了呢。
但是虽然在不同的栈里,在显示后台任务的时候 MainActivity 和 FragmentTestActivity在同一个窗口里,所以最近任务的显示单位并不是任务栈,而是taskAffinity。
关于应用场景
singleTask 大家都使用的比较多,比如首页,很多场景都希望在跳转到首页的时候把原来在它之上的都清理掉,singleTask 就很合适。
singleInstance 很多人都会说如果这个页面会被其他App调起的时候,这个页面就应该用singleInstance 但是我用singleTask的时候也完全没有问题,当跨应用调起Activity的时候,singleTask的Activity也会出现在一个新栈里。
其实只想到了一种情况,如果App在singleTask页面跳转了一个自己的页面,然后其他App再跳转到singleTask 的页面,那么这个自己的页面就会被销毁,singleInstance 就避免了这种情况。
还有就是singleTask 的Activity,按返回键的时候会回到第三方App的上个页面,其实我们想回到的是自己的页面。因为singleInstance只有一个元素,所以会回到我们自己的页面。
最后提一下FLAG_ACTIVITY_NEW_TASK
这个标志位的意思就是 如果目标App已经存在了Activity应该在的任务栈,那么就把Activity放到存在的任务栈里,如果没存在就创建新的任务栈。 应该在的任务栈指的是,如果设置了taskAffinity就是设置的,如果没有就是默认的。
这个FLAG 我感觉只有两个作用
一个是 当我们启动外部应用的Activity时,可以添加上这个FLAG,最近任务会显示你的我们的App和外部App两个任务,如果不加而且外部activity不是singleTask或者singleInstance的话,会只显示外部App的返回栈。
第二个是 使用非Activity 类型的 Context startActivity的时候,必须加上这个flag,有个解释是说 非Activity Context 本身就没有任务栈,所以加上这个flag的意思是让Activity自己去找一个任务栈,感觉也挺合理。
如果说有第三个的话 就是配合
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK or Intent.FLAG_ACTIVITY_CLEAR_TOP) 一块使用,有人说和singleTask作用一样,其实不是的,这个会创建新的Activity,但是singleTask不会。
费劲整理,如果有错误还请大家指正,感谢
网友评论