任务栈
每个应用都有一个任务栈,是用来存放Activity的,功能类似于函数调用的栈,先后顺序代表了Activity的出现顺序;比如Activity1-->Activity2-->Activity3,则任务栈为:
四种启动模式
- standard:每次激活Activity时(startActivity),都创建Activity实例,并放入任务栈;
- singleTop:如果该Activity位于任务栈栈顶,则不需要创建,复用实例并且会调用该实例的onNewIntent()方法将Intent对象传递到这个实例中,其余情况都要创建Activity实例;
- singleTask时,系统会把它标记为可在一个新任务中启动。至于启不启动,还要看具体情况(实质上就是:framework会将它的启动标志设为FLAG_ACTIVITY_NEW_TASK),
- 存在与该Activity的taskAffinity相同的Task:不存在实例,则新建Activity实例(位于栈顶!而且不会干掉其他的Activity)。如果存在则把当前实例中复用实例,并把它上面的一些家伙全部干掉,暴君啊!
- 不存在与该Activity的taskAffinity相同的task,那么将在新task启动,并创建实例。(较少出现)
- singleInstance:
当一个应用程序加载一个singleInstance模式的Activity时,如果该Activity没有被实例化,那么就重新创建一个Task,并入栈,如果已经被实例化,那么就调用该Activity的onNewIntent;
singleInstance的Activity所在的Task不允许存在其他Activity,任何从该Activity加载的其它Activity(假设为Activity2)都会被放入其它的Task中,如果存在与Activity2相同affinity的Task,则在该Task内创建Activity2。如果不存在,则重新生成新的Task并入栈。这个任务中必定只有这一个singleInstance类型的Activity
启动模式的应用及注意事项
- 启动模式和startActivityForResult()的关系:singleTask和singleInstance的Actvity,不允许用startActivityForResult启动!!
为什么通过startActivityForResult()方式去启动launchMode=singleTask的Activity,onActivityResult()会被立即回调且resultCode值为RESULT_CANCEL??
假设 singleTask 类型的Activity的taskAffinity属性设置的为另一个Task启动,并且这个Task不是空。那么它下面就还有其他Activity,当结束该singleTask 类型的Activity的时候,那么出现的将是另一个完全不相干的Activity而不是启动singleTask 类型的Activity。为了防止这种情况的发生。所以规定不许这么做。 - 启动模式,用来做安全退出或者闲置超时重新登录。利用singTask会实例复用时会把它上面的所有Activity都清除出栈的特性。所以singTask的Activity必须在Task根部。即使Activity被回收了,也仅仅处于被销毁状态。执行了onDestroy,没有执行finish,那么实例都还是存在栈里的。依然可以通过该方法实现安全退出。但是注意:如果该Activity被finish掉了,则无法实现清除上面的所有Activity的目的。因为它是新建实例重新入栈的,当转到该页面,只剩下这个页面在栈中,再finish掉该页面。就可以安全退出了。
- 也可以和广播退出想结合使用,实现完美退出方案。具体实施:如果我登录界面是singTask ,而且登录界面为根Activity,主页面也是singtask ,当我在主页面需要退出的时候。我就必须结束掉登录界面,发个广播就可以搞定。当然还有一个办法,那就是登录界面在转到主页面的时候就finish掉,让主页面做该Task的根Activity,但是这样就没法实现闲置超时的功能了。
因为登录界面之前被finish了。当再次转到登录界面。会在task的顶部重新new 个入栈。自然无法结束下面的应用。当你按回退键的时候又回去了,闲置退出也就失去了意义。
如有错误,欢迎指正,谢谢,如转载请在文章中明显注明出处