Activity四种启动模式

前言:

Activity的启动模式关系到Activity的缓存机制,正确的利用缓存机制有助于APP性能提升。

关于栈:

任务栈(Task Stack)
任务栈用来存放用户开启的Activity。
在应用程序创建之初,系统会默认分配给其一个任务栈(默认一个),并存储根Activity。
同一个Task Stack,只要不在栈顶,就是onStop状态。
 

1.默认启动模式standard

不设置启动模式,默认就是standard。在该模式下,启动的Activity会依次按照顺序压入Task中。会重复创建,如下图:

2.栈顶复用模式singleTop

在该模式下,如果栈顶Activity已经为我们要新建的Activity,不会重复创建。
应用场景:
开启渠道多,适合多应用开启调用Activity:通过这种设置可以避免已经创建过的Activity重复创建(多数通过动态设置使用)

3.栈内复用模式singleTask

与singleTop模式相似,只不过singleTop只针对栈顶元素,而singleTask模式下,如果task栈存在目标相同实例,则把该实例位置之上的所有实例都擦除掉,以保证当前实例在栈顶,如果不存在,则在栈顶创建该实例。
应用场景:
程序主界面,我们肯定不希望主界面被多创建,而且在主界面退出的时候退出整个App是最好的设想。
耗费系统资源的Activity:对于那些创建时及其耗费系统资源的Activity,我们可以考虑将其设为singleTask模式,减少GC回收的压力。

4.全局单例模式singleInstance

在该模式下,我们会为目标Activity分配一个新的Task栈,将目标Activity放入新的Task,并让目标Activity获得焦点,新的Task有且只有这一个Activity实例。如果已经创建过目标Activity实例,则不会创建新的Task,而是将以前创建过的Task唤醒。举例:比方 A Activity是该模式,启动A后。系统会为它创建一个单独的任务栈,由于栈内复用的特性。新的请求不会创建新的Activity,除非这个独特的任务栈被系统销毁。是一种加强的SingleTask模式。

启动模式的使用方式

1. 静态创建

在 Manifest.xml中指定Activity启动模式,这样在代码中跳转时会依照指定的模式来创建Activity。样例如下:
 <activity android:name="..activity.MultiportActivity" android:launchMode="singleTask"/>

2.动态创建

启动Activity时,在Intent的addFlags方法去指定启动模式,样例如以下:
Intent intent = new Intent();        
intent.setClass(context, MainActivity.class);        
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);        
context.startActivity(intent);

注意事项:
优先级:动态指定永远比静态优先级要高,若两者同时存在,以动态指定为准。
限定范围:静态无法为Activity指定 FLAG_ACTIVITY_CLEAR_TOP 标识,动态无法为Activity指定singleInstance模式。

启动模式:
介绍几个基本的标记位,切勿死记,理解几个就可以,须要时再查官方文档
Intent.FLAG_ACTIVITY_SINGLE_TOP
与静态设置中的singleTop效果相同
Intent.FLAG_ACTIVITY_CLEAR_TOP
Activity会检查Task中是否存在此实例,如果没有则添加压入栈,如果有,就将该Activity位置之上的所有Activity都擦除掉,此时有以下两种情况:
1.如果同时设置Intent.FLAG_ACTIVITY_SINGLE_TOP,则直接使用栈内的对应Activity
2.如果没有设置,将栈内的对应Activity销毁再重新创建。
准确的说,在Intent.FLAG_ACTIVITY_SINGLE_TOP和Intent.FLAG_ACTIVITY_CLEAR_TOP同时设置的情况下,相当于singleTask模式。
唯一不同的一点在于:同时设置会销毁已存在的目标实例,再重新创建。
FLAG_ACTIVITY_NEW_TASK
如果设置,此活动将成为此历史记录堆栈上新任务的开始。
FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS
具有此标记位的Activity不会出如今历史Activity的列表中,使用场景:当某些情况下我们不希望用户通过历史列表回到Activity时,此标记位便体现了它的效果。
它等同于在xml中指定Activity的属性:android:excludeFromRecents="trure"

如果不起作用,可以试试这个的组合:
Intent.FLAG_ACTIVITY_CLEAR_TASK | Intent.FLAG_ACTIVITY_NEW_TASK
或者在跳转Activity以后,当前Activity调用finish();

FLAG_ACTIVITY_NO_HISTORY

官方部分介绍:
FLAG_ACTIVITY_CLEAR_TASK
如果在传递给的Intent中设置Context#startActivity,则此标志将导致在活动开始之前清除与该活动相关的任何现有任务。

int	FLAG_ACTIVITY_CLEAR_TOP
如果已设置,并且正在启动的活动已经在当前任务中运行,那么与其启动该活动的新实例,不如关闭该活动之上的所有其他活动,并将此Intent传递给(现在顶部)将旧活动作为新的Intent。

int	FLAG_ACTIVITY_CLEAR_WHEN_TASK_RESET
在API级别21中不赞成使用此常数。从API 21开始,此常数与FLAG_ACTIVITY_NEW_DOCUMENT应使用的常数相同 。

int	FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS
如果设置,则新活动不会保留在最近启动的活动列表中。

int	FLAG_ACTIVITY_FORWARD_RESULT
如果已设置并且此意图用于从现有活动启动新活动,则现有活动的答复目标将转移到新活动。

int	FLAG_ACTIVITY_LAUNCHED_FROM_HISTORY
此标志通常不是由应用程序代码设置的,而是由历史记录(longpress主页键)启动的,由系统为您设置。

int	FLAG_ACTIVITY_LAUNCH_ADJACENT
此标志仅在分屏多窗口模式下使用。

int	FLAG_ACTIVITY_MATCH_EXTERNAL
如果在传递给的Intent中设置,Context#startActivity则如果设备上没有完整的应用程序可以处理该Intent ,则此标志将尝试启动即时应用程序。

int	FLAG_ACTIVITY_MULTIPLE_TASK
此标志用于创建新任务并向其中启动活动。

int	FLAG_ACTIVITY_NEW_DOCUMENT
此标志用于将文档打开到基于此Intent启动的活动的新任务中。

int	FLAG_ACTIVITY_NEW_TASK
如果设置,此活动将成为此历史记录堆栈上新任务的开始。

int	FLAG_ACTIVITY_NO_ANIMATION
如果在传递给的Intent中设置Context#startActivity,则此标志将阻止系统应用活动过渡动画进入下一个活动状态。

int	FLAG_ACTIVITY_NO_HISTORY
如果设置,则新活动不会保留在历史记录堆栈中。

int	FLAG_ACTIVITY_NO_USER_ACTION
如果设置,此标志将防止在Activity.onUserLeaveHint() 将新启动的活动移到最前面的活动暂停之前在当前最前面的活动上发生常规回调。

int	FLAG_ACTIVITY_PREVIOUS_IS_TOP
如果已设置并且此意图用于从现有活动中启动新活动,则当前活动将不被视为决定是否应将新意图传递到顶部而不是开始新意图的顶级活动。

int	FLAG_ACTIVITY_REORDER_TO_FRONT
如果在传递给的Intent中设置Context#startActivity,则此标志将使已启动的活动(如果已在运行)被带到其任务的历史记录堆栈的最前面。

int	FLAG_ACTIVITY_RESET_TASK_IF_NEEDED
如果已设置,并且此活动是在新任务中启动或将现有任务置于顶部,则它将作为任务的前门启动。

int	FLAG_ACTIVITY_RETAIN_IN_RECENTS
默认情况下,FLAG_ACTIVITY_NEW_DOCUMENT当用户关闭文档时,由创建者创建的文档将删除其在近期任务中的条目(带后盖,否则可能会完成())。

int	FLAG_ACTIVITY_SINGLE_TOP
如果设置了该活动,则如果该活动已经在历史记录堆栈的顶部运行,则不会启动该活动。

int	FLAG_ACTIVITY_TASK_ON_HOME
如果在传递给的Intent中设置Context#startActivity,则此标志将导致新启动的任务放置在当前家庭活动任务的顶部(如果有)。

官方文档

如果设置,则新活动不会保留在历史记录堆栈中。

点击Home键后再次打开导致APP重启问题:

在启动Activity里的OnCreate里加入以下方法
//首次启动 Intent.FLAG_ACTIVITY_BROUGHT_TO_FRONT 为 0,再次点击图标启动时就不为零了
        if ((getIntent().getFlags() & Intent.FLAG_ACTIVITY_BROUGHT_TO_FRONT) != 0) {
            finish();
            return;
        }

//也可以这样改,避免从桌面启动程序后,会重新实例化入口类的activity
        if (!this.isTaskRoot()) {
            Intent intent = getIntent();
            if (intent != null) {
                String action = intent.getAction();
                if (intent.hasCategory(Intent.CATEGORY_LAUNCHER) && Intent.ACTION_MAIN.equals(action)) {
                    finish();
                    return;
                }
            }
        }

这两种方法在 setContentView() 方法之前和之后都可以。

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值