Activity详解(下)

一.Activity四种启动模式

 

1.知识补充

任务是用户在执行某项工作时与之互动的一系列 Activity 的集合。Activity的管理是采用任务栈的形式。任务栈采用“后进先出”的栈结构。每按一次Back键,就有一个Activity出栈 。

 

 

 

 

2.启动模式的设置

 

启动模式有两种设置方式

 

<1> 通过 AndroidMainifest 设置

 <activity
     android:name=".activity.MainActivity"
     android:screenOrientation="portrait"
     android:launchMode="standard"/>

android:launchMode可取值以及含义

   (1) standard:标准模式

         默认值。系统在启动该 Activity 的任务中创建 Activity 的新实例,即启动Activity和目标Activity在同一Activity栈中。Activity 可以多次实例化,一个任务可以拥有多个实例

 

   (2) singleTop栈顶复用模式         

         如果当前任务的顶部已存在 目标Activity 的实例,则系统会通过调用其 onNewIntent() 方法来将 intent 转送给该实例,而不是创建 Activity 的新实例。Activity 可以多次实例化,一个任务可以拥有多个实例(但前提是返回堆栈顶部的 Activity 不是该 Activity 的现有实例)。

         例如,假设任务的返回堆栈包含根 Activity A 以及 Activity B、C 和位于顶部的 D(堆栈为 A-B-C-D;D 位于顶部)。收到以 D 类型 Activity 为目标的 intent。如果 D 采用默认的 "standard" 启动模式,则会启动该类的新实例,并且堆栈将变为 A-B-C-D-D。但是,如果 D 的启动模式为 "singleTop",则 D 的现有实例会通过onNewIntent() 方法接收 intent,因为它位于堆栈顶部,堆栈仍为 A-B-C-D。但是,如果收到以 B 类型 Activity 为目标的 intent,则会在堆栈中添加 B 的新实例,并且堆栈将变为 A-B-C-D-B,即使其启动模式为 "singleTop" 也是如此。

 

   (3) singleTask栈内复用模式

         系统会创建新任务,并实例化目标Activity。但是,如果任务中已存在该 Activity 的实例,则系统会通过调用其onNewIntent() 方法将 intent 转送到该现有实例,而不是创建新实例。Activity一次只能有一个实例存在

 

 

   (4) singleInstance:单例模式。

         与 "singleTask" 相似,唯一不同的是系统不会将任何其他 Activity 启动到包含该实例的任务中。该 Activity 始终是其任务唯一的成员;由该 Activity 启动的任何 Activity 都会在其他的任务中打开。

 

 

 

补充:在AndroidMainifest设置launchMode时,几个配套使用的属性

 

 

 

 

 

<2> 通过 Intent 设置标志位

Intent intent=new Intent(SplashActivity.this, MyActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(intent);

 

addFlags可取值以及含义

   (1) FLAG_ACTIVITY_NEW_TASK         

        在新任务中启动 Activity。如果您现在启动的 Activity 已经有任务在运行,则系统会将该任务转到前台并恢复其最后的状态,而 Activity 将在onNewIntent() 方法中收到新的 intent。

       launchMode属性"singleTask" 产生的行为相同。

 

 

   (2) FLAG_ACTIVITY_CLEAR_TOP

        如果要启动的 Activity 已经在当前任务中运行,则不会启动该 Activity 的新实例,而是会销毁位于它之上的所有其他 Activity,并通过onNewIntent() 方法将此 intent 传送给它的已恢复实例(现在位于堆栈顶部)。

      launchMode属性没有可产生此行为的值。

  FLAG_ACTIVITY_CLEAR_TOP 最常与 FLAG_ACTIVITY_NEW_TASK 结合使用。将这两个标记结合使用,可以查找其他任务中的现有 Activity,并将其置于能够响应 intent 的位置。

 

 

   (3) FLAG_ACTIVITY_SINGLE_TOP       

        如果要启动的 Activity 是当前 Activity(即位于返回堆栈顶部的 Activity),则现有实例会收到对onNewIntent() 方法的调用,而不会创建 Activity 的新实例。

         launchMode属性 "singleTop" 产生的行为相同。

 

 

<3> 二者设置的区别

有些启动模式可通过清单文件定义,但不能通过 intent 标记定义,同样,有些启动模式可通过 intent 标记定义,却不能在清单中定义。

                                  

 

 

 

3.亲和性

 

<1> 简介

亲和性表示Activity倾向于属于哪个任务默认情况下,同一应用中的所有 Activity 彼此具有亲和性。因此,在默认情况下,同一应用中的所有 Activity 都倾向于位于同一任务

不过,也可以修改Activity的默认亲和性。在不同应用中定义的Activity可以具有相同的亲和性,或者在同一应用中定义的 Activity 也可以被指定不同的任务亲和性。

您可以使用 <activity> 元素的 taskAffinity 属性修改任何给定 Activity 的亲和性。

taskAffinity 属性采用字符串值,该值必须不同于<manifest>元素中声明的默认软件包名称,因为系统使用该名称来标识应用的默认任务亲和性。

 

 

 

<2> 作用

亲和性可在两种情况下发挥作用

(1) 当启动 Activity 的 intent 包含 FLAG_ACTIVITY_NEW_TASK 标记时。

   默认情况下,新 Activity 会启动到调用 startActivity() 的 Activity 的任务中。它会被推送到调用方 Activity 所在的返回堆栈中。但是,如果传递给 startActivity() 的 intent 包含 FLAG_ACTIVITY_NEW_TASK 标记,则系统会寻找其他任务来容纳新 Activity。通常会是一个新任务,但也可能不是。如果已存在与新 Activity 具有相同亲和性的现有任务,则会将 Activity 启动到该任务中。如果不存在,则会启动一个新任务。

   如果此标记导致 Activity 启动一个新任务,而用户按下主屏幕按钮离开该任务,则必须为用户提供某种方式来返回到该任务。有些实体(例如通知管理器)总是在外部任务中启动 Activity,而不在它们自己的任务中启动,因此它们总是将 FLAG_ACTIVITY_NEW_TASK 添加到传递给 startActivity() 的 intent 中。如果您的 Activity 可由外部实体调用,而该实体可能使用此标记,请注意用户可以通过一种独立的方式返回到所启动的任务,例如使用启动器图标(任务的根 Activity 具有一个 CATEGORY_LAUNCHER intent 过滤器;请参阅下面的启动任务部分)。

 

(2) 当 Activity 的 allowTaskReparenting 属性设为 "true" 时。

   在这种情况下,一旦和 Activity 有亲和性的任务进入前台运行,Activity 就可从其启动的任务转移到该任务。

   举例来说,假设一款旅行应用中定义了一个报告特定城市天气状况的 Activity。该 Activity 与同一应用中的其他 Activity 具有相同的亲和性(默认应用亲和性),并通过此属性支持重新归属。当您的某个 Activity 启动该天气预报 Activity 时,该天气预报 Activity 最初会和您的 Activity 同属于一个任务。不过,当旅行应用的任务进入前台运行时,该天气预报 Activity 就会被重新分配给该任务并显示在其中。

 

 

 

4.举例说明

描述

MainActivity 跳转到AAAActivity,然后跳转到BBBActivity,再跳转到CCCActivity,最后跳转到DDDActivity。然后由DDDActivity跳转到BBBActivity。

 

 

<1> 所有Activity的启动模式都是默认模式。

完成所有跳转 查看Activity栈信息(Terminal 输入 adb shell dumpsys activity activities)

 

由此可见Activity栈信息为:MainActivity—AAA—BBB—CCC—DDD。在此基础上再启动BBB,虽然BBB已经存在在Activity栈中,可由于BBB的启动模式是标准模式,所以继续新建其实例放于Activity栈顶。此时Activity栈变成 MainActivity—AAA—BBB—CCC—DDD—BBB

 

 

 

 

<2> BBBActivity的启动模式变成singleTop。

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

完成所有跳转 查看Activity栈信息(Terminal 输入 adb shell dumpsys activity activities)

 

由此可见Activity栈信息为:MainActivity—AAA—BBB—CCC—DDD。在此基础上再启动BBB,虽然BBB已经存在在Activity栈中,可由于BBB的启动模式是singleTop 并且BBB的实例不在栈顶,所以继续新建其实例放于Activity栈顶。此时Activity栈变成 MainActivity—AAA—BBB—CCC—DDD—BBB。

 

 

 

<3> BBBActivity的启动模式变成singleTask。

<activity
    android:name=".mode.BBBActivity"
    android:launchMode="singleTask"/>

完成所有跳转 查看Activity栈信息(Terminal 输入 adb shell dumpsys activity activities)

 

 

BBBActivity

package com.wjn.lubandemo.mode;

import android.content.Intent;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.View;
import android.widget.TextView;

import com.wjn.lubandemo.R;

public class BBBActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_bbb);

        Log.d("BBBActivity", "onCreate方法执行");

        TextView textView = findViewById(R.id.textview2);
        textView.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent intent = new Intent(BBBActivity.this, CCCActivity.class);
                startActivity(intent);
            }
        });

    }

    @Override
    protected void onNewIntent(Intent intent) {
        super.onNewIntent(intent);
        getIntentvalue(intent);
    }

    /**
     * 获取Intent传值
     */

    private void getIntentvalue(Intent intent) {
        if (null != intent) {
            String name = intent.getStringExtra("name");
            int age = intent.getIntExtra("age", 0);
            Log.d("BBBActivity", "onNewIntent方法执行name----:" + name);
            Log.d("BBBActivity", "onNewIntent方法执行age----:" + age);
        }
    }

}

 

DDDActivity

package com.wjn.lubandemo.mode;

import android.content.Intent;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.TextView;

import com.wjn.lubandemo.R;

public class DDDActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_ddd);

        TextView textView = findViewById(R.id.textview4);
        textView.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent intent = new Intent(DDDActivity.this, BBBActivity.class);
                intent.putExtra("name","张三");
                intent.putExtra("age",30);
                startActivity(intent);
            }
        });
    }
}

Log日志

D/BBBActivity: onNewIntent方法执行name----:张三

D/BBBActivity: onNewIntent方法执行age----:30

 

由此可见Activity栈信息为:MainActivity—AAA—BBB—CCC—DDD。在此基础上再启动BBB,BBB已经存在在Activity栈中且BBB的启动模式是singleTask,所以不会新建其实例,而是将Activity栈中位于BBB之上的Activity全部销毁使BBB位于栈顶,并执行BBB的onNewIntent方法。此时Activity栈变成 MainActivity—AAA—BBB。

 

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值