zoukankan      html  css  js  c++  java
  • Android启动Activity的两种方式与四种启动模式*

    1、在一个Activity中调用startActivity()方法

    2、在一个Activity中调用startActivityRequest()方法。

    重写onActivityResult方法,用来接收B回传的数据。在B中回传数据时采用setResult方法,并且之后要调用finish方法。

    第一种方法简单直接。但是如果A调用B,并传递数据,同时B对数据处理后又返回给A,A再将数据显示出来。碰到这种情况,用第一种方法需要在A的onCreate()里面判断是第一次生成的界面,还是由B打开的A。这样比较麻烦,用第二种方法就简单了,在A的onCreate()只用写一次生成的界面的内容。在A的onActivityResult方法里放B处理完数据后的内容就可以了。

    下面是第二中方法的具体讲解:

    MainActivity 中部分代码
        Intent intent = new Intent(MainActivity.this, ActivityA.class);
        startActivity(intent);
    Intent intent = new Intent(MainActivity.this, ActivityB.class); startActivityForResult(intent, RequestCodeB);
    Intent intent = new Intent(MainActivity.this, ActivityC.class); startActivityForResult(intent, RequestCodeC); /******************************************************************************** * Activity回调 *******************************************************************************/ public static final int RequestCodeA = 10001; public static final int RequestCodeB = 10002; public static final int RequestCodeC = 10003; protected void onActivityResult(int requestCode, int resultCode, Intent data) { super.onActivityResult(requestCode, resultCode, data); if (requestCode == RequestCodeA && resultCode == RESULT_OK) { Log.d("MainActivity", "onActivityResultA"); } else if (requestCode == RequestCodeB && resultCode == RESULT_OK) { Log.d("MainActivity", "onActivityResultB"); } else if (requestCode == RequestCodeC && resultCode == RESULT_OK) { String string=data.getStringExtra("data"); Log.d("MainActivity", "onActivityResultC:"+string); } };
    public class ActivityA extends Activity {
    
        @Override
        public boolean onKeyDown(int keyCode, KeyEvent event) {
            if (keyCode == KeyEvent.KEYCODE_BACK) {
                finish();
            }
            return false;
        }
    }
    public class ActivityB extends Activity {
        @Override
        public boolean onKeyDown(int keyCode, KeyEvent event) {
            if (keyCode == KeyEvent.KEYCODE_BACK) {
                setResult(Activity.RESULT_OK);
                finish();
            }
            return false;
        }
    }
    public class ActivityC extends Activity {
        @Override
        public boolean onKeyDown(int keyCode, KeyEvent event) {
            if (keyCode == KeyEvent.KEYCODE_BACK) {
                Intent data = new Intent();
                data.putExtra("data", "ActivityC");
                setResult(Activity.RESULT_OK,data);
                finish();
            }
            return false;
        }
    }

    ***************************************************************************************

    Activity启动方式有四种,分别是:

    standard
    singleTop
    singleTask
    singleInstance

    可以根据实际的需求为Activity设置对应的启动模式,从而可以避免创建大量重复的Activity等问题。

    设置Activity的启动模式,只需要在AndroidManifest.xml里对应的<activity>标签设置android:launchMode属性,例如:

    <activity  
        android:name=".A1"  
        android:launchMode="standard" />  

    下面是这四种模式的作用:
    standard
    默认模式,可以不用写配置。在这个模式下,都会默认创建一个新的实例。因此,在这种模式下,可以有多个相同的实例,也允许多个相同Activity叠加。
    例如:
    若我有一个Activity名为A1, 上面有一个按钮可跳转到A1。那么如果我点击按钮,便会新启一个Activity A1叠在刚才的A1之上,再点击,又会再新启一个在它之上……
    点back键会依照栈顺序依次退出。
    singleTop
    可以有多个实例,但是不允许多个相同Activity叠加。即,如果Activity在栈顶的时候,启动相同的Activity,不会创建新的实例,而会调用其onNewIntent方法。
    例如:
    若我有两个Activity名为B1,B2,两个Activity内容功能完全相同,都有两个按钮可以跳到B1或者B2,唯一不同的是B1为standard,B2为singleTop。
    若我意图打开的顺序为B1->B2->B2,则实际打开的顺序为B1->B2(后一次意图打开B2,实际只调用了前一个的onNewIntent方法)
    若我意图打开的顺序为B1->B2->B1->B2,则实际打开的顺序与意图的一致,为B1->B2->B1->B2。
    singleTask
    只有一个实例。在同一个应用程序中启动他的时候,若Activity不存在,则会在当前task创建一个新的实例,若存在,则会把task中在其之上的其它Activity destory掉并调用它的onNewIntent方法。
    如果是在别的应用程序中启动它,则会新建一个task,并在该task中启动这个Activity,singleTask允许别的Activity与其在一个task中共存,也就是说,如果我在这个singleTask的实例中再打开新的Activity,这个新的Activity还是会在singleTask的实例的task中。
    例如:
    若我的应用程序中有三个Activity,C1,C2,C3,三个Activity可互相启动,其中C2为singleTask模式,那么,无论我在这个程序中如何点击启动,如:C1->C2->C3->C2->C3->C1-C2,C1,C3可能存在多个实例,但是C2只会存在一个,并且这三个Activity都在同一个task里面。
    但是C1->C2->C3->C2->C3->C1-C2,这样的操作过程实际应该是如下这样的,因为singleTask会把task中在其之上的其它Activity destory掉。
    操作:C1->C2          C1->C2->C3          C1->C2->C3->C2            C1->C2->C3->C2->C3->C1             C1->C2->C3->C2->C3->C1-C2
    实际:C1->C2          C1->C2->C3          C1->C2                              C1->C2->C3->C1                               C1->C2
    若是别的应用程序打开C2,则会新启一个task。
    如别的应用Other中有一个activity,taskId为200,从它打开C2,则C2的taskIdI不会为200,例如C2的taskId为201,那么再从C2打开C1、C3,则C2、C3的taskId仍为201。
    注意:如果此时你点击home,然后再打开Other,发现这时显示的肯定会是Other应用中的内容,而不会是我们应用中的C1 C2 C3中的其中一个。
    singleInstance
    只有一个实例,并且这个实例独立运行在一个task中,这个task只有这个实例,不允许有别的Activity存在。
    例如:
    程序有三个ActivityD1,D2,D3,三个Activity可互相启动,其中D2为singleInstance模式。那么程序从D1开始运行,假设D1的taskId为200,那么从D1启动D2时,D2会新启动一个task,即D2与D1不在一个task中运行。假设D2的taskId为201,再从D2启动D3时,D3的taskId为200,也就是说它被压到了D1启动的任务栈中。
    若是在别的应用程序打开D2,假设Other的taskId为200,打开D2,D2会新建一个task运行,假设它的taskId为201,那么如果这时再从D2启动D1或者D3,则又会再创建一个task,因此,若操作步骤为other->D2->D1,这过程就涉及到了3个task了。

    在IntentActivity中重写下列方法:onCreate onStart onRestart  onResume  onPause onStop onDestroy  onNewIntent

    1、其他应用发Intent,执行下列方法:
    onCreate
    onStart
    onResume

    发Intent的方法:

    Uri uri = Uri.parse("philn://blog.163.com");
    Intent it = new Intent(Intent.ACTION_VIEW, uri);   
    startActivity(it);

    2、接收Intent声明:

    复制代码
    <activity android:name=".IntentActivity" android:launchMode="singleTask"
      android:label="@string/testname">
          <intent-filter>
              <action android:name="android.intent.action.VIEW" />
                    <category android:name="android.intent.category.DEFAULT" />
                    <category android:name="android.intent.category.BROWSABLE" />
                    <data android:scheme="philn"/>
          </intent-filter>
    </activity>
    复制代码

    3、如果IntentActivity处于任务栈的顶端,也就是说之前打开过的Activity,现在处于onPause、onStop状态的话,其他应用再发送Intent的话,执行顺序为:
    onNewIntent,onRestart,onStart,onResume。

    Android应用程序开发的时候,从一个Activity启动另一个Activity并传递一些数据到新的Activity上非常简单,但是当您需要让后台运行的Activity回到前台并传递一些数据可能就会存在一点点小问题。

    首先,在默认情况下,当您通过Intent启到一个Activity的时候,就算已经存在一个相同的正在运行的Activity,系统都会创建一个新的Activity实例并显示出来。为了不让Activity实例化多次,我们需要通过在AndroidManifest.xml配置activity的加载方式(launchMode)以实现单任务模式,如下所示:

    <activity android:label="@string/app_name" android:launchmode="singleTask"android:name="Activity1"></activity>

    launchMode为singleTask的时候,通过Intent启到一个Activity,如果系统已经存在一个实例,系统就会将请求发送到这个实例上,但这个时候,系统就不会再调用通常情况下我们处理请求数据的onCreate方法,而是调用onNewIntent方法,如下所示:

    复制代码
    protected void onNewIntent(Intent intent) {
      super.onNewIntent(intent);
      setIntent(intent);//must store the new intent unless getIntent() will return the old one
      processExtraData();
    }
    复制代码

    不要忘记,系统可能会随时杀掉后台运行的 Activity ,如果这一切发生,那么系统就会调用 onCreate 方法,而不调用 onNewIntent 方法,一个好的解决方法就是在 onCreate 和 onNewIntent 方法中调用同一个处理数据的方法,如下所示:

    复制代码
    public void onCreate(Bundle savedInstanceState) {
          super.onCreate(savedInstanceState);     
          setContentView(R.layout.main);     
          processExtraData();     
    }
    protected void onNewIntent(Intent intent) {     
          super.onNewIntent(intent);     
           setIntent(intent);
          processExtraData()     
    }
    private void processExtraData(){     
          Intent intent = getIntent();     
          //use the data received here     
    }
     
    复制代码

    二、onNewIntent()的setIntent()和getIntent()

    复制代码
    @Override
    protected void onNewIntent(Intent intent) {
        super.onNewIntent(intent);     
        // setIntent(intent);     
         int data = getIntent().getIntExtra("HAHA", 0);
        // int data = intent.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()的时候就不会有问题了。

  • 相关阅读:
    BZOJ 3924: [Zjoi2015]幻想乡战略游戏
    codevs 4244 平衡树练习
    BZOJ 2002: [Hnoi2010]Bounce 弹飞绵羊
    BZOJ 2038: [2009国家集训队]小Z的袜子
    luogu P3709 大爷的字符串题
    BZOJ 2120: 数颜色
    luogu P2056 采花
    luogu P2709 小B的询问
    BZOJ 1878: [SDOI2009]HH的项链
    Codeforces 221d D. Little Elephant and Array
  • 原文地址:https://www.cnblogs.com/chenxibobo/p/6136626.html
Copyright © 2011-2022 走看看