理解 Activity 的启动模式
- Activity 的启动模式有哪几种,分别用于什么场景?
目前有四种启动模式:standard(标准模式)、singleTop(栈顶单例模式)、singleTask(栈内单例模式)、singleInstance(单独任务栈模式)。
应用场景:
- standard:Activity 默认使用的是 standard,一般配置界面使用这种模式
- singleTop:在任务栈内开启此模式下的 Activity,会进入栈顶,不会创建新的 Activity 实例。适合接收通知启动的内容显示页面。例如,某个新闻客户端的新闻内容页面,如果收到10个新闻推送,每次都打开一个新闻内容页面是很烦人的,这样的话可以从外界可能多次跳转到一个界面。
- singleTask:在任务栈内开启此模式下的 Activity,会 kill 掉此 Activity 上边的所有实例并进入栈顶。此模式的 Activity 适用于程序入口点,比如,我的实际应用中,程序开启后监听 usb 的状态,如果有 usb 设备挂载,程序会询问是否开启这个程序,点击是,会重新开启此程序,此时程序会有两个首页在运行,而我们有一个入口就可以了,所以可以采用 singleTask 。
- singleInstance:在任务栈内开启此模式下的 Activity,会创建一个新的任务栈,这个任务栈中只有此模式下的 Activity 。在做一些特殊场景下的应用,比如Launcher的主屏时,可能会使用得到。
- 清晰地描述下 onNewIntent 和 onConfigurationChanged 这两个生命周期方法的场景?
为了在日志中更好识别是哪个类的生命周期,我多加了几个最后一个字母
- onNewIntent
-
singleTop
设置 ActivityA 的 launchMode 为 singleTop ,首次启动 ActivityA 会执行
11-30 17:07:08.698 onCreate: ActivityAAAAAAAAAAAA
11-30 17:07:08.699 onStart: ActivityAAAAAAAAAAAA
11-30 17:07:08.701 onResume: ActivityAAAAAAAAAAAA
当 ActivityA 处于栈顶时,再次打开 ActivityA 后,会执行
11-30 17:07:10.568 onPause: ActivityAAAAAAAAAAAA
11-30 17:07:10.568 onNewIntent: ActivityAAAAAAAAAAAA
11-30 17:07:10.569 onResume: ActivityAAAAAAAAAAAA
首先失去焦点然后回调 onNewIntent 方法,并没有重建 Activity -
singleTask
设置 ActivityA 的 launchMode 为 singleTask 执行如下操作
ActivityA -> ActivityB -> ActivityA
ActivityA启动
11-30 17:18:07.714 onCreate: ActivityAAAAAAAAAAAA
11-30 17:18:07.714 onStart: ActivityAAAAAAAAAAAA
11-30 17:18:07.716 onResume: ActivityAAAAAAAAAAAA
ActivityB启动,此时ActivityA处于栈底,ActivityB处于栈顶
11-30 17:25:53.169 onPause: ActivityAAAAAAAAAAAA
11-30 17:25:53.202 onCreate: ActivityBBBBBBBBBBBB
11-30 17:25:53.212 onStart: ActivityBBBBBBBBBBBB
11-30 17:25:53.213 onResume: ActivityBBBBBBBBBBBB
11-30 17:25:53.627 onSaveInstanceState: ActivityAAAAAAAAAAAA
11-30 17:25:53.628 onStop: ActivityAAAAAAAAAAAA
再启动ActivityA
11-30 17:27:34.814 onPause: ActivityBBBBBBBBBBBB
11-30 17:27:34.827 onNewIntent: ActivityAAAAAAAAAAAA
11-30 17:27:34.829 onRestart: ActivityAAAAAAAAAAAA
11-30 17:27:34.830 onStart: ActivityAAAAAAAAAAAA
11-30 17:27:34.830 onResume: ActivityAAAAAAAAAAAA
11-30 17:27:35.173 onStop: ActivityBBBBBBBBBBBB
11-30 17:27:35.174 onDestroy: ActivityBBBBBBBBBBBB
最后一步, ActivityA 重建但是并没有执行 onCreate 方法,替代的是 onNewIntent 方法。 -
singleInstance
设置 ActivityA 的 launchMode 为 singleInstance 执行如下操作
ActivityA -> ActivityB -> ActivityA
前两步同上,我们直接看第三步的执行流程,此时 ActivityA 处于栈底, ActivityB 处于栈顶
11-30 17:39:56.956 onPause: ActivityBBBBBBBBBBBB 11-30 17:39:56.988 onNewIntent: ActivityAAAAAAAAAAAA 11-30 17:39:56.991 onRestart: ActivityAAAAAAAAAAAA 11-30 17:39:56.991 onStart: ActivityAAAAAAAAAAAA 11-30 17:39:56.992 onResume: ActivityAAAAAAAAAAAA 11-30 17:39:57.775 onSaveInstanceState: ActivityBBBBBBBBBBBB 11-30 17:39:57.777 onStop: ActivityBBBBBBBBBBBB
可以看出区别在于ActivityB并没有执行onDestroy方法,也就是说ActivityB并没有销毁
此时按Back键返回,ActivityB回到了前台
11-30 17:50:36.034 onPause: ActivityAAAAAAAAAAAA 11-30 17:50:36.088 onRestart: ActivityBBBBBBBBBBBB 11-30 17:50:36.089 onStart: ActivityBBBBBBBBBBBB 11-30 17:50:36.090 onResume: ActivityBBBBBBBBBBBB 11-30 17:50:36.851 onStop: ActivityAAAAAAAAAAAA 11-30 17:50:36.852 onDestroy: ActivityAAAAAAAAAAAA
此时ActivityA已销毁,再次按Back键则返回到主界面
- onConfigurationChanged
-
设置config为orientation|keyboardHidden|screenSize后切横竖屏android:configChanges="orientation|keyboardHidden|screenSize"
11-30 17:59:46.901 onConfigurationChanged: ActivityAAAAAAAAAAAA
ActivityA 不会销毁再重建,而是直接走 onConfigurationChanged,可用于需要横竖屏切换的场景。