15 - Activity四种加载模式之SingleTop

来源:互联网 发布:亿网域名证书 编辑:程序博客网 时间:2024/05/17 12:02
转载注明出处:http://blog.csdn.net/eana_don/
资料参考:官网API
通常情况下的Activity,默认是standard模式,同一个activity不断调用也是创建不同的实例。Task用来保存Activity,以栈的形式列队,先进后出。当一个Activity启动时,就会被压入task栈。而当用户返回时,则该Activity会被弹出栈,以这种形式来达到返回之前Activity的效果。但有些情况下,这种栈的形式不满足于我们的需求。比如,同一个activity都是同样的东西,有时候需A->B->A,并且不希望A重新创建实例浪费资源。此时,我们就需要设法打破默认的堆栈的形式。打破默认形式有两种方法,一为在AndroidManifest.xml定义Activity时指定它的加载模式。一为是在用Intent开启一个Activity时,在Intent中加入标志。这里重点于前者。
Activity的四种加载模式:standard,singletop,singletask,singleinstance。standard的应用频繁,很容易被理解。所以不累赘多述。这里主要是singletop。其他的两种查看博客不同的博文。


一、原理

SingleTop是指在栈顶仅保留一个同样的activity。
 即:如果SingleTopActivity在栈顶,就不会再重新创建一个新的同样的实例。 如果SingleTopActivity不在栈顶(可能存在,可能不存在,只要栈顶没有),就会再重新创建一个新的同样的实例。已有实例的情况下不会再启动新的实例,那假如同一个实例中,收到的数据发生了改变,因为没有重新oncreate,所以不会收到In'ten't,那怎么收到数据呢?这时候需要重写:onNewIntent()
所以分为两种情况:
1.SingleTopActivity在栈顶,比如A->A->A->A,实际上是什么都没做,相当于没有启动新的Activity。所以按返回时直接退出程序。
2.SingleTopActivity在栈顶,A为singletop模式,比如A->B->A,实际上是创建了A的实例,再创建了B的实例,再创建了A的实例(因为A不在栈顶)。返回的时候就是B->A。


原理图:


二、设置方式

设置方式很简单的,只要在AndroidManifest.xml加入如下即可:

 <activity android:name=".SingleTopActivityMain" android:launchMode="singleTop"></activity>


三、代码示例

MainActivity.java

package com.example.lunchmode;import android.app.Activity;import android.content.Intent;import android.os.Bundle;import android.view.Menu;import android.view.View;import android.view.View.OnClickListener;import android.widget.Button;public class MainActivity extends Activity {private Button singleTopBtn = null;private Button singleTaskBtn = null;private Button singleInstanceBtn = null;    @Override    public void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        System.out.println("   类名:" +this.getClass().getName() +" been created!");        setContentView(R.layout.activity_main);        singleTopBtn = (Button) findViewById(R.id.btn_singletop);        singleTopBtn.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View v) {// TODO Auto-generated method stubIntent intent = new Intent(MainActivity.this,SingleTopActivityMain.class);startActivity(intent);}});        singleTaskBtn = (Button) findViewById(R.id.btn_singletask);        singleTaskBtn.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View v) {// TODO Auto-generated method stubIntent intent = new Intent(MainActivity.this,SingleTaskActivityMain.class);startActivity(intent);}});              singleInstanceBtn = (Button) findViewById(R.id.btn_singleinstance);        singleInstanceBtn.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View v) {// TODO Auto-generated method stubIntent intent = new Intent(MainActivity.this,SingleInstanceActivityMain.class);startActivity(intent);}});    }    @Override    public boolean onCreateOptionsMenu(Menu menu) {        getMenuInflater().inflate(R.menu.activity_main, menu);        return true;    }    @Overrideprotected void onDestroy() {// TODO Auto-generated method stubSystem.out.println("   类名:" +this.getClass().getName() +" been destroyed!");super.onDestroy();}    }

activity_main.java

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"    xmlns:tools="http://schemas.android.com/tools"    android:layout_width="match_parent"    android:layout_height="match_parent"     android:orientation="vertical">    <Button        android:id="@+id/btn_singletop"        android:layout_width="fill_parent"        android:layout_height="wrap_content"        android:text="SingleTop" />     <Button        android:id="@+id/btn_singletask"        android:layout_width="fill_parent"        android:layout_height="wrap_content"        android:text="SingleTask" />     <Button        android:id="@+id/btn_singleinstance"        android:layout_width="fill_parent"        android:layout_height="wrap_content"        android:text="SingleInstanse" /></LinearLayout>


SingleTopActivityMain.java

package com.example.lunchmode;import android.app.Activity;import android.content.Intent;import android.os.Bundle;import android.view.Menu;import android.view.View;import android.view.View.OnClickListener;import android.widget.Button;import android.widget.TextView;/** * SingleTop是指在栈顶仅保留一个同样的activity。 * 即:如果SingleTopActivity在栈顶,就不会再重新创建一个新的同样的实例 *    如果SingleTopActivity不在栈顶,就会再重新创建一个新的同样的实例 * */public class SingleTopActivityMain extends Activity {private TextView tvHint = null;private Button btnNext = null;    @Override    public void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_content);        tvHint = (TextView) findViewById(R.id.tv_hint);        btnNext = (Button) findViewById(R.id.btn_next);        tvHint.setText("   类名:" +this.getClass().getName()+"   地址:" +this.toString());        //如果SingleTopActivity在栈顶,就不会再重新创建一个实例        btnNext.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View v) {// TODO Auto-generated method stubIntent intent = new Intent(SingleTopActivityMain.this,SingleTopActivityMain.class);startActivity(intent);}});              //如果SingleTopActivity不在栈顶,就会再重新创建一个实例//        btnNext.setOnClickListener(new OnClickListener() {////@Override//public void onClick(View v) {//// TODO Auto-generated method stub//Intent intent = new Intent(SingleTopActivityMain.this,MainActivity.class);//startActivity(intent);//}//});            }    @Override    public boolean onCreateOptionsMenu(Menu menu) {        getMenuInflater().inflate(R.menu.activity_main, menu);        return true;    }    }

activity_content.xml

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"    xmlns:tools="http://schemas.android.com/tools"    android:layout_width="match_parent"    android:layout_height="match_parent"     android:orientation="vertical">    <TextView         android:id="@+id/tv_hint"        android:layout_width="fill_parent"        android:layout_height="wrap_content"        android:text="A"/>    <Button        android:id="@+id/btn_next"        android:layout_width="fill_parent"        android:layout_height="wrap_content"        android:text="Next" />    <Button        android:id="@+id/btn_home"        android:layout_width="fill_parent"        android:layout_height="wrap_content"        android:text="Home" /></LinearLayout>