在Android中每个界面都是一个Activity,切换界面操作其实是多个不同Activity之间的实例化操作。在Android中Activity的启动模式决定了Activity的启动运行方式。
Android总Activity的启动模式分为四种:
Activity启动模式设置:
<activity android:name=".MainActivity"android:launchMode="standard" />
Activity的四种启动模式:
1.standard
模式启动模式,每次激活Activity时都会创建Activity,并放入任务栈中。
2.singleTop
如果在任务的栈顶正好存在该Activity的实例,就重用该实例,并且会调用实例的OnNewIntent()方法,否者就会创建新的实例并放入栈顶(即使栈中已经存在该Activity实例,只要不在栈顶,都会创建实例)。
3.singleTask
如果在栈中已经有该Activity的实例,就重用该实例(会调用实例的onNewIntent())。重用时,会让该实例回到栈顶,因此在它上面的实例将会被移除栈。如果栈中不存在该实例,将会创建新的实例放入栈中。
4.singleInstance
在一个新栈中创建该Activity实例,并让多个应用共享改栈中的该Activity实例。一旦改模式的Activity的实例存在于某个栈中,任何应用再激活改Activity时都会重用该栈中的实例,其效果相当于多个应用程序共享一个应用,不管谁激活该Activity都会进入同一个应用中。
其中standard是系统默认的启动模式。
初始界面如下:
下面通过实例来演示standard的运行机制:
当点击按钮时,会创建新的Activity,通过TextView@后16进制数的显示即可看出,点击两次分别界面如下:
三次显示的内容都不一样,而且点击返回键以后会一次返回到上一级Activity。
此时,我们分析栈内部的运行机制:
(依次从栈顶向上)
因此,这种Standard模式是每次都会创建新的Activity对象,当点击返回按钮时,他会将栈顶(当前Activity)消灭,然后跳到前一层,例如如果现在Activity是40540100,那么当我们点击返回时Activity会变为40538d10,不过此时在这个Activity中再次点击按钮创建对象时,它会另外创建新的Activity对象,这种模式可能大多数情况下不是我们需要的,因为对系统性能的消耗过大。
举个例子:
1、假如你从主页进入一个页面中浏览信息如果里边有个链接是返回主页,如果你的页面是用standard模式,那么就会新创建一个主页的对象,当你返回的时候你就会返回到上一级浏览信息的界面,再返回会返回到上一级界面也就是主页,如果你在里边进行了多次操作,那么就会产生很多垃圾对象,占用大量的内存空间。
2、假如你的页面中有一个按钮或者链接能再次启动当前页面,那么你就会再次创建一个当前页面的对象,这样就造成了对象的重复创建。
平时我们对于这种事件的解决都是在启动Intnet后把当前不需要的页面给finish()掉,这样可以解决。但是也就多写了一行代码,可能会影响效率。
2. singleTop
从上面的解释中即可知道,在每次使用新的Activity时会自动检测栈顶的当前Activity是否是需要引用的Activity,如果是则直接引用此Activity,而不会创建新的Activity。
我们在刚才的界面中加入一个"启动singletop模式"按钮,当点击时出现我们创建的singletop中,在Activity singletop中有一个按钮,启动singletop模式,表示启动当前Activity,由于我们在清单文件中配置Activity的启动模式为singleTop,因此此时不会再创建而是利用当前栈顶的singleTopActivity:
<activity
android:name=".SingleTopActivity"
android:launchMode="singleTop" />
初始化:
点击"启动singleTop模式"按钮
我们分析它的运行机制,可知,当程序运行到此时,栈中的数据形式为:
虽然它不会创建新的Activity对象,不过它每次会调用onNewIntent()方法:
@Override
protected void onNewIntent(Intent intent) {
// TODO Auto-generated method stub
super.onNewIntent(intent);
Toast.makeText(this, new Date().toString(), 1).show();
}
我们为此方法编写代码输出当前日期,则在每次点击上面按钮时会输出当前日期。
这种模式就解决了在当前页面下依然启动自身页面而创建多余对象的问题,每次启动都会判断需要启动的页面是否是在栈的最顶端也就是当前页面,如果是就调用已有的对象不在创建重复的对象。
3. singleTask
此启动模式和singleTop在名字上即可看出区别,即singleTop每次只检测当前栈顶的Activity是否是我们需要请求创建的,而singleTask则会检测栈中全部的Activity对象,从上向下,如果检测到是我们所请求的则会消灭此Activity对象上面的对象,直接把检测到的我们需要的Activity置为栈顶。
我们创建一个SingleTaskActivity,此界面中包含一个启动MainActivity和启动SingleTaskActivity按钮。
初始化:
点击"启动singleTask模式"按钮:
在此界面中点击第二个按钮"启动singleTask模式"按钮,根据定义会检测当前栈中是否有此Activity对象,因此显示的还是当前的Activity,不会重新创建;
再点击"启动Standard模式"按钮,由于MainActivity的启动模式为standard,所以在此会重新创建一个MainActivity对象:
再次点击启动Standard模式:
此时栈中数据格式为:
这种模式就挺适合在多次操作后,如果想要回到最开始的页面,那么启动以后不但能回到那个页面,而且这其中所有的操作页面均销毁。减少了垃圾对象占用的内存空间。
4.SingleInstance
此启动模式和我们使用的浏览器工作原理类似,我们都知道在多个程序中访问浏览器时,如果当前浏览器没有打开,则打开浏览器,否则会在当前打开的浏览器中访问。此模式会节省大量的系统资源,因为他能保证要请求的Activity对象在当前的栈中只存在一个。
1)全局单实例,该Activity在销毁前只能被创建一次
2)在任何地方启动该Activity,如果该Activity未创建,则直接在新的task中创建该activity,如果以创建,则将其所在的task带回前台。执行onNewIntent方法
3)该Activity不与任何其它activity存在于同一task中
OnNewIntent():
该方法是activity的一个保护类型的方法,利用已有的Acivity去处理别的Intent时,你就可以利用onNewIntent来处理,通常被用在有搜索请求的activity,而其该activity有好几个intent-filter,该方法被调用的前提
a、该activity设置如下属性android:launchMode="singleTop | singleTask",。
b、该activity已经处在栈的顶端,通过其他的方法又重新启动该acitvity时被调用,如搜索,这时oncreate()方法不调用。