注册

大厂Android高频问题:谈谈Activity的启动模式?

前言

Activity可以说是Android开发面试高频的一道问题,但总有小伙伴在回答这道问题总不能让面试满意, 在这你就要搞清楚面试问你对Activity启动模式时,他最想听到的和其实想问的应该是哪些?下面我们通过以下几点来剖析这道问题!

  1. 启动模式是什么?
  2. 启动模式如何设置?
  3. Activity的启动模式区别?
  4. 应用场景以及哪些注意的点?

1.activity堆栈流程以及四种启动模式

一个应用由多个Activity构成,多个Activity构成了任务,系统以栈方式进行管理任务(也就是管理多个Activity),管理方式为“先进后出”。

默认情况下,当用户点击App图标后,启动应用,这时会创建一个任务栈,并且将MAIN Activity压入栈中,作为栈底Activity。之后每启动一个Activity,就会将这个Activity压入栈中,显示处于栈顶的Activity。当用户点击“返回”键后,处于栈顶的Activity进行出栈销毁。

Android提供四种Activity的启动模式来进行入栈操作。

standard:

默认值,启动Activity都会重新创建一个Activity的实例进行入栈。此时Activity可能存在多个实例。

image.png

singleTop:

当Activity处于栈顶时,再启动此Activity,不会重新创建实例入栈,而是会使用已存在的实例。

image.png

singleTask:

与singleTop模式相似,只不过singleTop模式是只是针对栈顶的元素,而singleTask模式下,如果task栈内存在目标Activity实例,则:

  1. 将task内的对应Activity实例之上的所有Activity弹出栈。
  2. 将对应Activity置于栈顶,获得焦点。

image.png

singleInstance:

这是我们最后的一种启动模式,也是我们最恶心的一种模式:在该模式下,我们会为目标Activity分配一个新的affinity,并创建一个新的Task栈,将目标Activity放入新的Task,并让目标Activity获得焦点。新的Task有且只有这一个Activity实例。

如果已经创建过目标Activity实例,则不会创建新的Task,而是将以前创建过的Activity唤醒(对应Task设为Foreground状态)

image.png

2.启动模式如何设置?

AndroidMainfest.xml文件设置

设置的lanuchMode属性。可设置四个值: standard、singleTop、singleTask、singleInstance。若不设置默认为standard。

<activity 
android:name=".activity.MainActivity"
android:launchMode="standard"/>

Intent跳转标记Flag

FLAG_ACTIVITY_SINGLE_TOP 等价于 singleTop。位于栈顶的Activity会重用实例,调用onNewIntent函数接收intent。

Intent intent = new Intent(this, MainActivity.class);
intent.setFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP);
startActivity(intent);

FLAG_ACTIVITY_SINGLE_NEW_TASK,启动新的TASK,这个新的TASK取决于xml中设置的TaskAffinity(亲和性)属性。

首先去寻找是否存在相同亲和性的任务,如果存在,那么直接将这个Activity加入到这个任务中。若不存在,则新建一个任务来加入Activity。

FLAG_ACTIVITY_CLEAR_TOP,会将位于此Activity上方的Activity进行出栈销毁。

// singleTask的行为可使用代码表示为
Intent intent = new Intent(this, MainActivity.class);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TOP);
startActivity(intent);

两者区别:

  • xml设置为静态的
  • intent标记是动态的。intent标记Flag的优先级更高一些。所以当标记Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TOP后,尽管Activity为默认的standard模式,也同样会使用存在的实例,调用onNewIntent。

3.亲和性和多个任务并存

亲和性是指Activity设置在AndroidMainfest.xml中的taskAffinity属性。相同亲和性的Activity在同一个任务中,默认使用application的taskAffinity,也就是package name。

并不是设置taskAffinity就一定起作用,起作用是有条件的:

  • 同时设置了launchMode属性为singleTask。
  • Intent跳转时使用FLAG_ACTIVITY_NEW_TASK。
  • 同时设置allowTaskReparenting属性为true。
  • allowTaskReparenting可以使此Activity从启动任务中转移到该taskAffinity的任务中。此时需要发生Task reset(回到Home之后再进入app)才能看出效果。

不同亲和性意味着不同的任务,也就是同一个app中可以存在不同的任务,前台显示的任务的栈顶Activity为用户可见的Activity。当启动一个新的任务时,新的任务会覆盖当前任务。并且回退时,一个任务中Activity全部出栈,会将后台的任务调出,直到最后一任务的最后一个Activity出栈,app结束,回到Home。

例如: 一个应用有Main、A、B、C四个Activity,C的lanuchMode为singleTask,并且taskAffinity设置为.c,其他都为默认,那么按照启动顺序: Main->A->B->C

此时存在两个task:

  • 默认: Main->A->B (后台)
  • .c: C (前台)

由C启动A,那么此时task为:

  • 默认: Main->A->B->A (前台)

  • .c: C (后台)

按回退键:

  • 出栈顺序为: A、B、A、Main、C

4.应用场景以及需要避免的坑

  1. 新闻客户端的推送,点击打开新闻详情页,此时新闻详情页应该设置singleTop,避免用户在新闻详情页打开推送通知,使得回退出现两次详情页。

  2. 利用singleTask的特性,可以使得应用完全退出。

    • 注意:闪屏页+主页+其他的应用,可以设置主页为singleTask,因为闪屏页展示完就finish掉,栈底存在主页,用户点击回退键可以直接关闭应用。

    • 坑:避免启动MAIN的MainActivity设置为singleTask,这样当用户点击HOME,再重新启动应用时,将始终展示MainActivity,并且此时MainActivity走onNewIntent方法。

  3. singleInstance使用比较少,系统应用比如打电话可使用singleInstance。

  4. singleTop、singleTask、singleInstance在使用已存在的Activity实例时,都将走onNewIntent方法。

看完三件事❤️

如果你觉得这篇内容对你还蛮有帮助,我想邀请你帮我三个小忙:

  1. 点赞,转发,有你们的 『点赞和评论』,才是我创造的动力。
  2. 关注公众号 『 小新聊Android 』,不定期分享原创知识
  3. 同时可以期待后续文章ing🚀

0 个评论

要回复文章请先登录注册