一、android中Activity的启动模式
在android中,每个界面都是一个Activity,切换界面其实就是不同Activity之间的实例化操作,在android中Activity的启动模式决定了Activity的启动运行方式。
android中activity的四种启动方式:
Activity启动模式设置:
<activity android:name=".MainActivity" android:launchMode="standard" />
Activity的四种启动模式:
1. standard
默认启动模式,每次激活Activity时都会创建Activity,并放入任务栈中。
2. singleTop
如果在任务的栈顶正好存在该Activity的实例, 就重用该实例,否者就会创建新的实例并放入栈顶(即使栈中已经存在该Activity实例,只要不在栈顶,都会创建实例)。
3. singleTask
如果在栈中已经有该Activity的实例,就重用该实例(会调用实例的onNewIntent())。重用时,会让该实例回到栈顶,因此在它上面的实例将会被移除栈。如果栈中不存在该实例,将会创建新的实例放入栈中。
4. singleInstance
在一个新栈中创建该Activity实例,并让多个应用共享改栈中的该Activity实例。一旦改模式的Activity的实例存在于某个栈中,任何应用再激活改Activity时都会重用该栈中的实例,其效果相当于多个应用程序共享一个应用,不管谁激活该Activity都会进入同一个应用中
一个应用的Activity供多种方式调用启动的情况,多个调用希望只有一个Activity的实例存在,这就需要Activity的onNewIntent(Intent intent)方法了。只要在Activity中加入自己的onNewIntent(intent)的实现加上Manifest中对Activity设置lanuchMode=“singleTask”就可以。
二、onNewIntent()
在IntentActivity中重写下列方法:onCreate onStart onRestart onResume onPause onStop onDestroy onNewIntent
1、当Activity第一次创建时,执行以下生命周期函数:onCreate()-->onStart()-->onResume()
2、接收Intent声明:
1 |
< activity android:name = ".IntentActivity" android:launchMode = "singleTask"
|
2 |
android:label = "@string/testname" >
|
4 |
< action android:name = "android.intent.action.VIEW" />
|
5 |
< category android:name = "android.intent.category.DEFAULT" />
|
6 |
< category android:name = "android.intent.category.BROWSABLE" />
|
7 |
< data android:scheme = "philn" />
|
3、如果IntentActivity处于任务栈的顶端,也就是说之前打开过的Activity,现在处于onPause、onStop状态的话,其他应用再发送Intent的话,执行顺序为:
onNewIntent()-->onRestart()-->onStart()-->onResume()。
在Android应用程序开发的时候,从一个Activity启动另一个Activity并传递一些数据到新的Activity上非常简单,但是当您需要让后台运行的Activity回到前台并传递一些数据可能就会存在一点点小问题。
首先,在默认情况下,当您通过Intent启到一个Activity的时候,就算已经存在一个相同的正在运行的Activity,系统都会创建一个新的Activity实例并显示出来。为了不让Activity实例化多次,我们需要通过在AndroidManifest.xml配置activity的加载方式(launchMode)以实现单任务模式,如下所示:
1 |
< activity android:label = "@string/app_name" android:launchmode = "singleTask" android:name = "Activity1" ></ activity >
|
launchMode为singleTask的时候,通过Intent启到一个Activity,如果系统已经存在一个实例,系统就会将请求发送到这个实例上,但这个时候,系统就不会再调用通常情况下我们处理请求数据的onCreate方法,而是调用onNewIntent方法,如下所示:
1 |
protected void onNewIntent(Intent intent) {
|
3 |
super .onNewIntent(intent);
|
不要忘记,系统可能会随时杀掉后台运行的
Activity
,如果这一切发生,那么系统就会调用
onCreate
方法,而不调用
onNewIntent
方法,一个好的解决方法就是在
onCreate
和
onNewIntent
方法中调用同一个处理数据的方法,如下所示:
1 |
public void onCreate(Bundle savedInstanceState) {
|
3 |
super .onCreate(savedInstanceState);
|
5 |
setContentView(R.layout.main);
|
1 |
protected void onNewIntent(Intent intent) {
|
3 |
super .onNewIntent(intent);
|
1 |
private void processExtraData(){
|
3 |
Intent intent = getIntent();
|
三、onNewIntent()的setIntent()和getIntent()
2 |
protected void onNewIntent(Intent intent) {
|
3 |
super .onNewIntent(intent);
|
7 |
int data = getIntent().getIntExtra( "HAHA" , 0 );
|
如果没有调用setIntent(intent),则getIntent()获取的数据将不是你所期望的。但是使用intent.getInXxx,貌似可以获得正确的结果。
注意这句话:
Note that getIntent() still returns the original Intent. You can use setIntent(Intent) to update it to this new Intent.
所以最好是调用setIntent(intent),这样在使用getIntent()的时候就不会有问题了。
四、友盟消息推送中的onNewIntent()使用方法。