上一篇 《Activity启动模式之详解ActivityRecord、TaskRecord、ActivityStack》记录了深入理解启动模式所需要的一系列相关类和名词,如果有需要可以先去看看 https://blog.csdn.net/luzaiyy611/article/details/99259950。
本篇尽量使用简洁的语言讲解Activity的四种启动模式以及一些需要注意的点以及简单的用法和使用场景。
启动模式的设计目的
启动模式的设计目的主要在于针对activity的实例进行复用,避免activity重复的创建多个实例,浪费资源以及实现一些符合正常交互逻辑的Activity跳转行为。
四种启动模式
1. Standard
Activity默认的标准启动模式,无需特意指定。
特性:每次启动该模式的activity会在app的activity栈顶总会新建一个该activity的实例。
2. SingleTop
栈顶复用模式。
特性:如果需要启动的activity实例已经处于所属任务栈顶部,那么直接复用该栈顶的activity,无需再次创建实例,并且回调onNewIntent方法;如果需要启动的activity实例不处于任务栈顶部或者不存在于任务栈中,那么新建该activity的实例置于任务栈顶部,同standard模式一样。
3. SingleTask
栈内复用模式。
特性:如果需要启动的目标activity实例已存在于所属任务栈中,那么直接复用该activity实例,无需创建新实例,回调onNewIntent方法,并且让所有在目标activity之上的activity实例出栈,使得目标activity位于当前task的栈顶;如果目标activity不存在于所属任务栈中,那么新建activity实例并将其放置于栈顶。
4. SingleInstance
全局唯一模式。
该模式具备singleTask模式的所有特性外,与它的区别就是,这种模式下的Activity会单独占用一个Task栈(如果stack中不存在该Task栈则新建一个Task栈并且只存放这个activity),具有全局唯一性,即整个系统中就这么一个实例,由于栈内复用的特性,后续的请求均不会创建新的Activity实例,除非这个特殊的任务栈被销毁了。以singleInstance模式启动的Activity在整个系统中是单例的,如果在启动这样的Activity时,已经存在了一个实例,那么会把它所在的任务调度到前台,重用这个实例。
使用方式
1. 在manifest中指定Activity的启动模式
<activity android:name=".MainActivity" android:launchMode="singleTask"/>
2. 启动Activity时,在Intent中指定相应的启动模式启动Activity
一种动态的启动模式,在new 一个Intent后,通过Intent的addFlags方法去动态指定一个启动模式。
关于Intent的各种flag由于篇幅问题本文不详细叙述。
Intent intent = new Intent();
intent.setClass(context, MainActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(intent);
区别
- 优先级 : 动态指定方式优先级高于Manifest指定方式,若二者同时出现,以动态指定方式为准。
- 限定范围 : Manifest指定方式无法直接指定FLAG_ACTIVITY_CLEAR_TOP 标识。
-
动态指定方式无法指定singleInstance模式。