Android TaskAffinity实践详解

介绍

其实一直以来都知道在AndroidManifest的Activity中可以配置TaskAffinity这个属性,只知道这个东西和Activity的任务栈有关,今天花了点时间好好学习了一些这个东西。

launchMode(启动模式)

Activity的任务栈是我们很熟悉的了,它是一种后进先出的结构。位于栈顶的Activity处于焦点状态,当按下back按钮的时候,栈内的Activity会一个一个的出栈,并且调用其onDestory()方法。如果栈内没有Activity,那么系统就会回收这个栈,每个APP默认只有一个栈,以APP的包名来命名.

  1. standard: 标准模式,每次启动Activity都会创建一个新的Activity实例,并且将其压入任务栈栈顶,而不管这个Activity是否已经存在。Activity的启动三回调(onCreate()->onStart()->onResume())都会执行。

  2. singleTop: 栈顶复用模式.这种模式下,如果新Activity已经位于任务栈的栈顶,那么此Activity不会被重新创建,所以它的启动三回调就不会执行,同时Activity的onNewIntent()方法会被回调.如果Activity已经存在但是不在栈顶,那么作用与standard模式一样.

  3. singleTask: 栈内复用模式.创建这样的Activity的时候,系统会先确认它所需任务栈已经创建,否则先创建任务栈.然后放入Activity,如果栈中已经有一个Activity实例,那么这个Activity就会被调到栈顶,onNewIntent(),并且singleTask会清理在当前Activity上面的所有Activity.(clear top)

  4. singleInstance: 加强版的singleTask模式,这种模式的Activity只能单独位于一个任务栈内,由于栈内复用的特性,后续请求均不会创建新的Activity,除非这个独特的任务栈被系统销毁了

TaskAffinity

好了,现在要开始介绍TaskAffinity了,如果单独设置TaskAffinity属性的话是没有任何效果的,只有Activity的launchMode设置成singTask的时候才会生效的。那么现在我们就来验证一下:

这里需要注意的是:
1.TaskAffinity的值应该是xxx.xxx.xxx类似包名的,如果没有包括.的话是安装不了的;  
2.如果不指定TaskAffinity的话,默认的值是包名。  
案例1

这里的MainActivity中的代码很简单,就是把当前的类名和TaskId打印出来,SecondActivity和ThirdActivity也是一样的。

输出:
cn.xdeveloper.activitytest.MainActivity
698

cn.xdeveloper.activitytest.SecondActivity
698

cn.xdeveloper.activitytest.ThirdActivity
698

这里可以看到这三个Activity的TaskId都相同,可见TaskAffinity属性并没有起作用。

案例2

修改SencondActivity的launchMode=singleTask。

输出:

cn.xdeveloper.activitytest.MainActivity
699

cn.xdeveloper.activitytest.SecondActivity
700

cn.xdeveloper.activitytest.ThirdActivity
700

这里可以看到SecondActivity的TaskId改变了,启动Activity的时候会根据taskAffinity查找是否有存在的任务栈,没有的话就创建一个新的任务栈。同时在启动ThirdActivity,也会加入到当前的任务栈中。

案例3

修改SencondActivity的launchMode=singleInstance,其他不变。

输出:

cn.xdeveloper.activitytest.MainActivity
703

cn.xdeveloper.activitytest.SecondActivity
704

cn.xdeveloper.activitytest.ThirdActivity
703

可以看到SecondActivity是创建在新的栈里,而ThirdActivity却还是创建在原来的栈里面,这是因为singleInstance的特性造成的,它会创建一个新的栈并且里边就只有一个Activity实例,所以和singTask不同,之后启动的ThirdActivity不会进入到该栈中。

其实这里SecondActivity中设置的taskAffinity是没意义的,就算不设置结果也是一样的,因为singleInstance会创建一个新的栈并只能保存唯一的Activity,所以其他的Activity就算设置了一样的taskAffinity也不起作用了。

这里讲一下启动过程:
MainActivity(703栈进入前台) --> SecondActivity(704栈进入前台,703栈进入后台) --> ThirdActivity(703栈进入前台,704栈进入后台)

最终栈里的情况是:
前台:703栈(ThirdActivity、MainActivity)
后台:702栈(SecondActivity)

所以这个时候依次点击返回键退回的话:
ThirdActivity --> MainActivity --> SecondActivity

总结

到这里TaskAffinity的作用已经很明了了,通过这个属性可以把不同的Activity分在不同的任务栈中,这里总结一下重点:

1.TaskAffinity属性只有在launchMode=singleTask的时候才有作用;(案例1、案例2)
2.正常情况下启动的Activity会默认加到当前的前台栈中;(案例2)