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被销毁重新创建
需求:禁用掉横竖屏切换的生命周期
- 横竖屏写死
android:screenOrientation="landscape"
android:screenOrientation="portrait"
- 让系统的环境 不再去敏感横竖屏的切换。
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);
- 在新开启的界面里面实现设置数据的逻辑
Intent data = new Intent();
data.putExtra("phone", phone);
//设置一个结果数据,数据会返回给调用者
setResult(0, data);
finish();//关闭掉当前的activity,才会返回数据
- 在开启者activity里面实现方法 onActivityResult(int requestCode, int resultCode, Intent data) 通过data获取返回的数据
- 根据请求码和结果码确定业务逻辑