Android Activity 启动模式
standard
默认启动模式,不需要在AndroidManifest.xml 中的文件声明,每次启动一个该模式下的Activity,都会创建一个Activity
singleTop
该模式下的Activity存在并处于Activity栈的栈顶的时候,当你启动创建该activity,它不会创建新的activity,而是会复用之前的activity
如果栈顶的Activity实例不是该Activity的实例对象,那么就会重新创建一个Activity的实例对象,将该实例对象添加栈中。
singleTop: ActivityA 启动 ActivityA, 就会复用之前ActivityA,并调用ActivityA的onNewIntent方法
启动Activity生命周期如下:onPause() --onNewIntent()–onResume()
singleTask
该模式下的Activity实例只有一个,在同一个程序内启动该Activity时,如果栈中没有该Activity的实例对象,那么会创建一个新的Activity对象,将对象添加到栈中,若栈中存在该Activity的实例,那么销毁掉在该Activity实例上的所有Activity实例,则该Activity实例处于栈顶位置,该实例的onNewIntent方法会被调用
singleInstance
该模式下的Activity实例只有一个,并且运行在一个独立的栈中,启动该Activity时,会创建一个新的栈,将该Activity对象添加到栈中
现象:当同时存在singleInstance和非singleInstance下的Activity时(未被销毁),切到后台,会发现2个Activity Task
Intent Flag标识:
FLAG_ACTIVITY_CLEAR_TOP:例如现在的栈情况为:A B C D 。D此时通过intent跳转到B,如果这个intent添加FLAG_ACTIVITY_CLEAR_TOP 标记,则栈情况变为:A B。如果没有添加这个标记,则栈情况将会变成:A B C D B。也就是说,如果添加了FLAG_ACTIVITY_CLEAR_TOP标记,并且目标Activity在栈中已经存在,则将会把位于该目标activity之上的activity从栈中弹出销毁。这跟Launch mode设置成singleTask类似,但是和SingleTask不一样的是,FLAG_ACTIVITY_CLEAR_TOP会把他上面的弹出,但是自身也销毁,然后重新创建个新对象,而SingleTask不会重新创建新对象,就是不会触发onCreate,只会触发onNewIntent
FLAG_ACTIVITY_NEW_TASK:例如现在栈1的情况是:A B C。C通过intent跳转到D,并且这个intent添加了FLAG_ACTIVITY_NEW_TASK 标记,如果D这个Activity在Manifest.xml中的声明中添加了Task affinity,并且和栈1的affinity不同,系统首先会查找有没有和D的Task affinity相同的task栈存在,如果有存在,将D压入那个栈,如果不存在则会新建一个D的affinity的栈将其压入。如果D的Task affinity默认没有设置,或者和栈1的affinity相同,则会把其压入栈1,变成:A B C D,这样就和不加FLAG_ACTIVITY_NEW_TASK 标记效果是一样的了。 注意如果试图从非activity的非正常途径启动一个activity,比如从一个service中启动一个activity,则intent比如要添加FLAG_ACTIVITY_NEW_TASK 标记。
FLAG_ACTIVITY_NO_HISTORY:例如现在栈情况为:A B C。C通过intent跳转到D,这个intent添加FLAG_ACTIVITY_NO_HISTORY标志,则此时界面显示D的内容,但是它并不会压入栈中。如果按返回键,返回到C,栈的情况还是:A B C。如果此时D中又跳转到E,栈的情况变为:A B C E,此时按返回键会回到C,因为D根本就没有被压入栈中。
FLAG_ACTIVITY_SINGLE_TOP:和上面Activity的 Launch mode的singleTop类似。如果某个intent添加了这个标志,并且这个intent的目标activity就是栈顶的activity,那么将不会新建一个实例压入栈中。
FLAG_ACTIVITY_NO_USER_ACTION
onUserLeaveHint()作为activity周期的一部分,它在activity因为用户要跳转到别的activity而要退到background时使用。比如,在用户按下Home键,它将被调用。比如有电话进来(不属于用户的选择),它就不会被调用。
那么系统如何区分让当前activity退到background时使用是用户的选择?
它是根据促使当前activity退到background的那个新启动的Activity的Intent里是否有FLAG_ACTIVITY_NO_USER_ACTION来确定的。
注意:调用finish()使该activity销毁时不会调用该函数 参考:这里
注意:当调用到onNewIntent(intent)的时候,需要在onNewIntent() 中使用setIntent(intent)赋值给Activity的Intent.否则,后续的getIntent()都是得到老的Intent。参考:这里
以下为补充:
startActivityForResult 启动activity:
1.某个Activity正在栈顶且为SingleTop模式,并且用startActivityForResult启动自己,此时SingleTop不起作用,被当作Standard模式,如果添加Intent.FLAG_ACTIVITY_NEW_TASK,则起作用,且onActivityResult会立即回调并收到RESULT_CANCEL。
2.ActivityA startActivityForResult 启动 ActivityB(SingleInstance),此时SingleInstance不起作用,被当作Standard模式。
3.ActivityA startActivityForResult 启动 ActivityB(Intent.FLAG_ACTIVITY_NEW_TASK+Standard),ActivityA的onActivityResult会立即回调并收到RESULT_CANCEL,Activity都在同一栈中,如果ActivityB为SingleInstance,则SingleInstance起作用,Activity位于不同栈。
参考:
[Activity生命周期以及launchMode对生命周期的影响]: https://blog.youkuaiyun.com/songshizhuyuan/article/details/47127969
[activity在singleTop模式下的生命周期]: https://blog.youkuaiyun.com/qq_33505655/article/details/80242327
[Android笔记(十四):分析SingleTask和SingleInstance启动模式]: https://blog.youkuaiyun.com/weixin_40855673/article/details/80630634
[关于SingleInstance使用的问题]: https://blog.youkuaiyun.com/Super_xiaojian/article/details/81027530
[Android启动模式之singleinstance的坑]: https://blog.youkuaiyun.com/mr_jingfu/article/details/79077613