Android应用开发-页面跳转和数据传递

Activity

Android四大组件之一 可以理解为Android中的界面,每一个界面都是一个Activity Activity的使用必须在清单文件中进行配置 在清单文件中,

创建第二个Activity

  • 需要在清单文件中为其配置一个activity标签
  • 标签中如果带有这个子节点,则会在系统中多创建一个快捷图标
<intent-filter>
     <action android:name="android.intent.action.MAIN" />
     <category android:name="android.intent.category.LAUNCHER" />
 </intent-filter>
  • 一个应用程序可以在桌面创建多个快捷图标。
  • activity的名称、图标可以和应用程序的名称、图标不相同
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
  • 栗子
<application
    android:allowBackup="true"
    android:icon="@drawable/ic_launcher"
    android:label="@string/app_name"
    android:theme="@style/AppTheme" >
     <!-- 第一个Activity配置清单 -->
    <activity
        android:icon="@drawable/f000"
        android:name="com.istarry.activity2.MainActivity"
        android:label="@string/app_name" >
//=========================================================================

    <intent-filter>
 <!-- 设置入口Activity,每个 <intent-filter>会生成一个新的图标 -->
        <action android:name="android.intent.action.MAIN" />

        <category android:name="android.intent.category.LAUNCHER" />
    </intent-filter>
</activity>
<!-- 第二个Activity配置清单 -->
<activity
    android:icon="@drawable/f000"
    android:name="com.istarry.activity2.SecondActivity"
    android:label="@string/app_name" >
<!--
    <intent-filter>
        <action android:name="android.intent.action.MAIN" />

        <category android:name="android.intent.category.LAUNCHER" />
    </intent-filter>
-->
</activity>

Activity跳转

  • 显式:指定目标Activity的包名和类名
  • 如果目标在本项目中:指定上下文和字节码
  • 如果目标Activity不在本项目中:指定包名和类名
  • 隐式:指定目标Activity的action和data
  • 通过设置intent对象中的属性,去跟目标Activity的<intent-filter>匹配
  • 一个Activity可以定义多个意图过滤器,隐式启动时只需要与任意一个匹配即可
  • 一个意图过滤器可以定义多个相同子节点,隐式启动时只需要与任意一个匹配即可
  • 显式启动比隐式效率高
  • 当目标Activity与启动者在同一个项目中,使用显式
  • 当目标Activity与启动者不在同一个项目中,使用隐式
  • 当隐式启动指定的Action与多个Activity匹配时,会弹出选择列表
/**
         启动拨打电话的Activity
         @param v
     */
    public void click(View v){
        //隐式意图
        Intent intent = new Intent();
        intent.setAction(Intent.ACTION_CALL);
        intent.setData(Uri.parse("tel:110"));
        //隐式跳转
        startActivity(intent);
    }

    public void click3(View v){
        //显式意图
        Intent intent = new Intent();
        //arg0:目标Activity所在的项目的应用包名
        //arg1:目标Activity的包名和类名
        intent.setClassName(packageContext, className);
    }

    /**
        启动第二个自定义Activity
     */
    public void click2(View v){
        //显式意图
        Intent intent = new Intent();
        //指定目标,设置参数1(上下文)参数2(指定目标Activity的字节码文件)
        intent.setClass(this, SecondActivity.class);
        //显式跳转
        startActivity(intent);
    }

隐式启动自定义Activity

/**
   * 隐式启动第二个自定义Activity
     */
public void click4(View v){
    //隐式意图
    Intent intent = new Intent();
    intent.setAction("com.istarry.suiyi");
    //隐式跳转
    startActivity(intent);
}
  • 隐式启动自定义Activity需要配置清单文件
<intent-filter>
    <action android:name="com.istarry.suiyi" />

    <category android:name="android.intent.category.DEFAULT" />
</intent-filter>
/**
     * 隐式启动第二个自定义Activity
     */
    public void click4(View v){
        //隐式意图
        Intent intent = new Intent();
        intent.setAction("com.istarry.suiyi");

        //设置数据
        intent.setData(Uri.parse("suiyixie:xxxxxx"));

        //如果没有指定Category,会自动添加Category
        intent.addCategory("android.intent.category.DEFAULT");

        //隐式跳转
        startActivity(intent);
    }
  • 清单文件
<intent-filter>
        <action android:name="com.istarry.suiyi" />

        <data android:scheme="suiyixie" />

        <category android:name="android.intent.category.DEFAULT" />
    </intent-filter>

---补充

//隐式意图
        Intent intent = new Intent();
        intent.setAction("com.istarry.suiyi");

        /*设置mimeType,必须与清单文件配置的一样
        intent.setType("text/name");

        //设置数据
        intent.setData(Uri.parse("suiyixie:xiaoming"));
         */
        //setType与setData不能共存,要用下面的方法
        intent.setDataAndType(Uri.parse("suiyixie:xiaoming"), "text/name");

        //如果没有指定Category,会自动添加Category
        intent.addCategory("android.intent.category.DEFAULT");
        //隐式跳转
        startActivity(intent);
  • 清单文件
<intent-filter>
        <action android:name="com.istarry.suiyi" />

        <data android:scheme="suiyixie" />
        <!-- 约定俗成,可以自定义mimeType -->
        <data android:mimeType="text/name"/>

        <category android:name="android.intent.category.DEFAULT" />
    </intent-filter>

Activity的生命周期

  • onCreate:创建时调用
  • onStart:此时Activity显示在屏幕上,但还未获得焦点
  • onResume:此时Activity已经获得焦点
  • onPause:此时Activity失去焦点,但依然可见
  • onStop:此时Activity已经不可见了
  • OnDestroy:此时Activity就死掉了
  • 使用场景
  • Activity创建时需要初始化资源,销毁时需要释放资源;或者播放器应用,在界面进入后台时需要自动暂停

如果应用的Activity进入stop和pause状态,在内存不足时,应用会被杀死,选择被杀死的目标是用 LRU(最近最少使用)算法

完整生命周期(entire lifetime)

onCreate-->onStart-->onResume-->onPause-->onStop-->onDestory

可视生命周期(visible lifetime)

onStart-->onResume-->onPause-->onStop

前台生命周期(foreground lifetime)

onResume-->onPause

Activity的四种启动模式

每个应用会有一个Activity任务栈,存放已启动的Activity

Activity的启动模式,修改任务栈的排列情况

  • standard 标准启动模式
  • singleTop 单一顶部模式
  • 如果任务栈的栈顶存在这个要开启的activity,不会重新的创建activity,而是复用已经存在的activity。保证栈顶如果存在,不会重复创建。
  • 应用场景:浏览器的书签
  • singeTask 单一任务栈,在当前任务栈里面只能有一个实例存在
  • 当开启activity的时候,就去检查在任务栈里面是否有实例已经存在,如果有实例存在就复用这个已经存在的activity,并且把这个activity上面的所有的别的activity都清空,复用这个已经存在的activity。保证整个任务栈里面只有一个实例存在
  • 应用场景:浏览器的activity
  • 如果一个activity的创建需要占用大量的系统资源(cpu,内存)一般配置这个activity为singletask的启动模式。webkit内核 c代码
  • singleInstance启动模式非常特殊, activity会运行在自己的任务栈里面,并且这个任务栈里面只有一个实例存在
  • 如果你要保证一个activity在整个手机操作系统里面只有一个实例存在,使用singleInstance
  • 应用场景: 电话拨打界面

横竖屏切换的生命周期

默认情况下 ,横竖屏切换, 销毁当前的activity,重新创建一个新的activity

快捷键ctrl+F11

在一些特殊的应用程序常见下,比如游戏,不希望横竖屏切换activity被销毁重新创建

需求:禁用掉横竖屏切换的生命周期

  1. 横竖屏写死
android:screenOrientation="landscape"
android:screenOrientation="portrait"
  1. 让系统的环境 不再去敏感横竖屏的切换。
android:configChanges="orientation|screenSize|keyboardHidden"

Activity跳转时的数据传递

  • Activity通过Intent启动时,可以通过Intent对象携带数据到目标Activity
Intent intent = new Intent(this, SecondActivity.class);
intent.putExtra("maleName", maleName);
intent.putExtra("femaleName", femaleName);
startActivity(intent);
  • 在目标Activity中取出数据
Intent intent = getIntent();
String maleName = intent.getStringExtra("maleName");
String femaleName = intent.getStringExtra("femaleName");

掌握开启activity获取返回值

从A界面打开B界面, B界面关闭的时候,返回一个数据给A界面

步骤: 1. 开启activity并且获取返回值

startActivityForResult(intent, 0);
  1. 在新开启的界面里面实现设置数据的逻辑
Intent data = new Intent();
data.putExtra("phone", phone);
//设置一个结果数据,数据会返回给调用者
setResult(0, data);
finish();//关闭掉当前的activity,才会返回数据
  1. 在开启者activity里面实现方法 onActivityResult(int requestCode, int resultCode, Intent data) 通过data获取返回的数据
  2. 根据请求码和结果码确定业务逻辑