今天要讲得是Activity的四种启动模式launchMode属性,该属性用于配置该Activity的加载模式,该属性支持以下4个属性值。
- standard:标准模式,也是默认模式。
- singletop:Task栈顶单例模式。
- singleTask:Task内单例模式。
- singlestance:全局单例模式。
在继续写下去之前我想说一下,像launchMode这种基础的Android知识在网上已经有很多大神写得十分详细,为什么我还要写呢?我之所以写,是因为我确确实实是个菜鸟,将课本、网上大神的总结再加上自己对知识点的认知去写出来希望让我对这个知识点更加了解。
以下是参考的博客:
- http://blog.csdn.net/liuhe688/article/details/6754323
- http://www.cnblogs.com/lwbqqyumidi/p/3771542.html
以下是参考的书籍:
- 《疯狂的Android讲义》电子工业出版社
回到话题,今天要弄懂的是:
- 为什么要为Activity指定加载模式?
- 每个加载模式有什么不同?
- 在什么时候应该用哪一种模式?
- 怎么样使用加载模式?
先介绍Android对Activity的管理,Android采用Task来管理多个Activity,当我们启动一个应用是,Android就会为其创建一个Task,然后启动这个应用的入口Activity (即 < intent-filter…/>)中配置的MAIN和LAUNCHER)。
可以把Task理解为任务栈,Task以栈的形式管理Activity:先入后出。通过调用Activity的getTaskId()的方法来获取他所在Task的ID。
launchMode在多个Activity跳转的过程中扮演着重要的角色,它可以决定是否生成新的Activity实例,是否重用已存在的Activity实例,是否和其他Activity实例共用一个Task。
那么Activity的加载模式就是负责管理实例化,加载Activity的方式,并可以控制Activity与Task之间的加载关系。
首先我们可以在AndroidManifest.xml中配置Activity的 android:launchMode属性:
<activity
android:name=".MainActivity"
android:launchMode="standard">
</activity>
接下来我们就开始认识这4种加载模式。
1.standard
每次通过这种模式启动目标Activity时,Android总会为其目标Activity创建一个新的实例,并将其添加到当前Task栈中,所有我们发现:是在同个任务栈里面,只是创建了不同的实例。
下面示例使用standard模式不断启动自身:
public class MainActivity3 extends ActionBarActivity {
private TextView tv;
private Button btn;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main_activity3);
tv = (TextView) findViewById(R.id.tv);//定义文本
btn = (Button) findViewById(R.id.Btn);//定义按钮
tv.setText(String.format("TaskId:%d\n Current Activity:%s", getTaskId(), toString()));
btn.setOnClickListener(new View.OnClickListener() {//为按钮添加监听器,当单击按钮时启动MainActivity3
@Override
public void onClick(View v) {
startActivity(new Intent(MainActivity3.this,MainActivity3.class));
}
});
}
正如以上代码,每次单击按钮都会启动自身MainActivity3,程序无需配置Activity的launchMode属性,因为默认是采用standard模式.
显然,当用户单击手机上面的”返回”键时,系统会“逐一”从Activity栈顶删除Activity实例。同时也表明这种加载模式不会使用全新的Task。
——————————
2.singleTop
我们先在AndroidManifest.xml中配置Activity的 android:launchMode属性:
<activity
android:name=".MainActivity3"
android:launchMode="singleTop">
</activity>
这种模式与standard模式基本相同,但有一点是不同:当将要启动的目标Activity已经位于Task栈顶时,系统不会重新创建目标Activity的实例。
使用的之前已修改启动模式的实例:
当单击按钮时,并没有什么反应。
为了更好的了解singleTop模式的作用,再创建一个Activity,MainActivity4。
现有MainActivity3、MainActivity4两个Activity实例,功能是1.启动自己,2.启动另一个Activity。就好比A启动A,A启动B;B启动A,B启动B。
代码只是多了个按钮 以及监听,这里就不写出来了
单击上面那个按钮就是A启动A,在singleTop模式中是没有反应的,当单击下面那个按钮时候就相当于 A启动B,此时我们来看看反应如何:
根据上图显示,我们以及跳转到了MainActivity4中,也就是我所说的B中, TaskId仍然与A中的12是一样的,但两个MainActivity实例是不一样的。
此时我们再单击第一个按钮,也就是说由B到A,
此时我们仔细看,MainActivity3的两次的实例都不一样。
就想之前所说 :当将要启动的目标Activity已经位于Task栈顶时,系统不会重新创建目标Activity的实例。而每次启动其他的目标Activity时,功能就像standard模式。
——————————
3.singleTask
先在AndroidManifest.xml中配置Activity的 android:launchMode属性:
<activity
android:name=".MainActivity3"
android:launchMode="singleTop">
</activity>
这个十分好理解,采用这种加载模式的Activity在同一个Task内只有一个实例,当系统采用singleTask模式启动目标Activity时,分为三种情况:
- 如果将要启动的目标Activity 不存在,系统将会创建目标Activity的实例,并加入Task任务栈顶。
- 如果将要启动的目标Activity 已经位于Task栈顶,此时与singleTop模式的行为相同。
- 如果要启动的Activity 已经存在、但没有位于Task栈顶,系统会将把位于该Activity上面所有Activity移出Task,从而使其转入栈顶 (大爷作风)。
——————————
4.singlelnstance
在AndroidManifest.xml中配置Activity的 android:launchMode属性:
<activity
android:name=".MainActivity3"
android:launchMode="singlelnstance">
</activity>
在这种加载模式下,系统保证无论从哪个Task中启动目标Activity,只会创建一个目标Activity实例,并会使用一个全新的Task栈来加载该Activity实例。
- 如果将要启动的目标Activity 不存在,系统会先创建一个全新的Task栈,再创建目标Activity实例,并将它假如新的Task栈顶。
- 如果将要启动的目标Activity 已经存在,无论它位于哪个应用程序中、位于哪个Task中,系统都会把该Activity所在的Task转到前台,从而使该Activity显示出来。
——————————
合抱之木,生於毫末;九層之台,起於累土;千里之行,始於足下。《老子》
作为一个程序员,树立远大的目标是值得欣赏的,但不能只去空想,要一步一步地实践才行。成大事者,须从小事做起;万事起于忽微,量变引起质变。 —— by @liuhe688