Android Studio开发实战:从零基础到App上线(第2版)
上QQ阅读APP看书,第一时间看更新

3.5 活动Activity基础

本节介绍Android四大组件之一Activity的基本概念和常见用法。首先说明Activity的生命周期,接着说明Intent的组成部分与工作原理,然后阐述如何使用Intent完成活动页面之间的消息传递,包括如何传递请求参数、如何返回应答参数等。

3.5.1 Activity的生命周期

看到这里,相信读者对Activity已经不陌生了。首先,一个Activity代表一个页面。其次,Activity的onCreate方法是页面的入口函数。更细心的读者也许已经知道调用startActivity方法可以跳转到下一个页面。之所以到这时才介绍Activity,是因为Activity的逻辑复杂、概念繁多,必须在有一定基础后讲解才合适,不然一开始就讲解高深的专业术语,读者恐怕很难理解。

首先介绍Activity的生命周期,如同花开花落一般,Activity也有从含苞待放到盛开再到凋零的生命过程。下面是Activity与生命周期有关的方法说明。

  • onCreate:创建页面。把页面上的各个元素加载到内存中。
  • onStart:开始页面。把页面显示在屏幕上。
  • onResume:恢复页面。让页面在屏幕上活动起来,例如开启动画、开始任务等。
  • onPause:暂停页面。让页面在屏幕上的动作停下来。
  • onStop:停止页面。把页面从屏幕上撤下来。
  • onDestroy:销毁页面。把页面从内存中清除掉。
  • onRestart:重启页面。重新加载内存中的页面数据。

下面针对几个常见的业务场景探究一下Activity的生命周期,主要有3个场景:页面之间的跳转、竖屏与横屏的切换、按HOME键与返回App。用于场景测试的代码如下,主要在每个生命周期函数中增加打印屏幕日志和后台日志。

图3-20 活动页面跳转时的界面日志截图

1. 页面之间的跳转

首先进入测试页面ActJumpActivity,接着从该页面跳转到ActNextActivity,然后从ActNextActivity返回ActJumpActivity。界面上的日志截图如图3-20所示。其中,区域1表示进入页面ActJumpActivity时的生命周期过程,区域2表示跳转到ActNextActivity时的生命周期过程,区域3表示返回ActJumpActivity时的生命周期过程。

从日志截图可以看到,下一个页面的创建伴随上一个页面的停止,不过显示的日志信息不够完整。下面跟踪一下logcat里的日志,看看这中间到底发生了什么。

首先打开页面ActJumpActivity,调用方法的顺序为:本页面onCreate→onStart→onResume。日志如下:

     11:30:18.352:D/ActJumpActivity(2315):onCreate
     11:30:18.352:D/ActJumpActivity(2315):onStart
     11:30:18.352:D/ActJumpActivity(2315):onResume

从ActJumpActivity跳转到ActNextActivity,调用方法的顺序为:上一个页面onPause→下一个页面onCreate→onStart→onResume→上一个页面onStop。日志如下:

     11:30:32.668:D/ActJumpActivity(2315):onPause
     11:30:32.688:D/ActNextActivity(2315):onCreate
     11:30:32.688:D/ActNextActivity(2315):onStart
     11:30:32.688:D/ActNextActivity(2315):onResume
     11:30:33.116:D/ActJumpActivity(2315):onStop

从ActNextActivity回到ActJumpActivity(按返回键或在代码中调用finish方法),调用的方法顺序为:下一个页面onPause→上一个页面onRestart→onStart→onResume→下一个页面onStop→onDestroy。日志如下:

     11:30:40.740:D/ActNextActivity(2315):onPause
     11:30:40.752:D/ActJumpActivity(2315):onRestart
     11:30:40.752:D/ActJumpActivity(2315):onStart
     11:30:40.752:D/ActJumpActivity(2315):onResume
     11:30:41.160:D/ActNextActivity(2315):onStop
     11:30:41.164:D/ActNextActivity(2315):onDestroy

图3-21 活动页面在横竖屏切换时的界面日志截

至此,基本上可以弄清楚页面跳转时的生命周期了。总体上是跳转前的页面先调用onPause方法,然后跳转后的页面依次调用onCreate/onRestart→onStart→onResume,最后跳转前的页面调用onStop方法(若返回上级页面,则下级页面还需调用onDestroy方法)。

2. 竖屏与横屏的切换

首先进入测试页面ActRotateActivity,此时默认为竖屏显示;接着倒转手机切换到横屏,观察日志;然后倒转手机切换回竖屏,观察日志。3个屏幕的显示日志时间没有重复,这里的日志截图是3次截图拼接而成的,如图3-21所示。

从日志截图可以看出,竖屏与横屏似乎在每次切换时页面都要重新创建。为进一步验证实验结果,再一次查看logcat里的日志信息如下:

     21:02:10.179 D/ActRotateActivity: onCreate
     21:02:10.179 D/ActRotateActivity: onStart
     21:02:10.179 D/ActRotateActivity: onResume
     21:02:13.227 D/ActRotateActivity: onPause
     21:02:13.227 D/ActRotateActivity: onStop
     21:02:13.227 D/ActRotateActivity: onDestroy
     21:02:13.247 D/ActRotateActivity: onCreate
     21:02:13.247 D/ActRotateActivity: onStart
     21:02:13.247 D/ActRotateActivity: onResume
     21:02:16.239 D/ActRotateActivity: onPause
     21:02:16.239 D/ActRotateActivity: onStop
     21:02:16.239 D/ActRotateActivity: onDestroy
     21:02:16.279 D/ActRotateActivity: onCreate
     21:02:16.279 D/ActRotateActivity: onStart
     21:02:16.279 D/ActRotateActivity: onResume

分析日志的时间与内容,无论是竖屏切换到横屏,还是横屏切换到竖屏,都是原屏幕的页面从onPause到onStop再到onDestroy一路销毁,然后新屏幕的页面从onCreate到onStart再到onResume一路创建而来。

3. 按HOME键与返回App

首先进入测试页面ActHomeActivity;接着按HOME键,屏幕回到桌面;然后按任务键或长按HOME键(不同手机的操作不一样),屏幕调出进程视图;最后点击测试App,屏幕返回测试页面。一路下来的屏幕日志截图如图3-22所示。

图3-22 按HOME键的界面日志截图

从日志截图可以看到,此时测试页面的生命周期是典型的从活动状态变为暂停状态(回到桌面时)再到活动状态(返回App页面时)。观察logcat的后台日志,发现后台日志与屏幕日志保持一致。

3.5.2 使用Intent传递消息

Intent的中文名是意图,意思是我想让你干什么,简单地说,就是传递消息。Intent是各个组件之间信息沟通的桥梁,既能在Activity之间沟通,又能在Activity与Service之间沟通,也能在Activity与Broadcast之间沟通。总而言之,Intent用于处理Android各组件之间的通信,完成的工作主要有3部分:

(1)Intent需标明本次通信请求从哪里来、到哪里去、要怎么走。

(2)发起方携带本次通信需要的数据内容,接收方对收到的Intent数据进行解包。

(3)如果发起方要求判断接收方的处理结果,Intent就要负责让接收方传回应答的数据内容。

为了做好以上工作,就要给Intent配上必须的装备,Intent的组成部分见表3-5。

表3-5 Intent组成元素的列表说明

表达Intent的来往路径有两种方式,一种是显式Intent,另一种是隐式Intent。

1. 显式Intent,直接指定来源类与目标类名,属于精确匹配。

在声明一个Intent对象时,需要指定两个参数,第一个参数表示跳转的来源页面,第二个参数表示接下来要跳转到的页面类。具体的声明方式有如下3种:

(1)在构造函数中指定,示例代码如下:

     Intent intent = new Intent(this, ActResponseActivity.class);  // 创建一个目标确定的意图

(2)调用setClass方法指定,示例代码如下:

     Intent intent = new Intent();  // 创建一个新意图
     intent.setClass(this, ActResponseActivity.class);  // 设置意图要跳转的活动类

(3)调用setComponent方法指定,示例代码如下:

     Intent intent = new Intent();  // 创建一个新意图
     ComponentName component = new ComponentName(this, ActResponseActivity.class);
     intent.setComponent(component);  // 设置意图携带的组件信息
2. 隐式Intent,没有明确指定要跳转的类名,只给出一个动作让系统匹配拥有相同字串定义的目标,属于模糊匹配。

因为我们常常不希望直接暴露源码的类名,只给出一个事先定义好的名称,这样大家约定俗成、按图索骥就好,所以隐式Intent起到了过滤作用。这个定义好的动作名称是一个字符串,可以是自己定义的动作,也可以是已有的系统动作。系统动作的取值说明见表3-6。

表3-6 系统动作的取值说明

这个动作名称通过setAction方法指定,也可以通过构造函数Intent(String action)直接生成Intent对象。当然,由于动作是模糊匹配,因此有时需要更详细的路径,比如知道某人住在天通苑小区,并不能直接找到他家,还得说明他住在天通苑的哪一期、哪号楼、哪一层、哪一个单元。Uri和Category便是这样的路径与门类信息,Uri数据可通过构造函数Intent(String action,Uri uri)在生成对象时一起指定,也可通过setData方法指定(setData这个名字有歧义,实际就是setUri);Category可通过addCategory方法指定,之所以用add而不用set方法,是因为一个Intent可同时设置多个Category,一起进行过滤。

下面是一个调用系统拨号程序的例子,其中就用到了Uri:

     Intent intent = new Intent();  // 创建一个新意图
     intent.setAction(Intent.ACTION_CALL);  // 设置意图动作为直接拨号
     Uri uri = Uri.parse("tel:" + phone);  // 声明一个拨号的Uri
     intent.setData(uri);  // 设置意图前往的路径
     startActivity(intent);  // 启动意图通往的活动页面

隐式Intent还用到了过滤器的概念,即把不符合匹配条件的过滤掉,剩下符合条件的按照优先顺序调用。创建一个Android工程,AndroidManifest.xml里的intent-filter就是XML中的过滤器。比如下面这个最常见的主页面MainAcitivity,activity节点下面便设置了action和category的过滤条件。其中,android.intent.action.MAIN表示App的入口动作,android.intent.category.LAUNCHER表示在App启动时调用。

3.5.3 向下一个Activity传递参数

前面讲过,Intent的setData方法只指定到达目标的路径,并非本次通信所携带的参数信息,真正的参数信息存放在Extras中。Intent重载了很多种putExtra方法传递各种类型的参数,包括String、int、double等基本数据类型,甚至Parcelable、Serializable等序列化结构。不过只是调用putExtra方法显然不好管理,像送快递一样大小包裹随便扔,不但找起来不方便,丢了也难以知道。所以Android引入了Bundle概念,可以把Bundle理解为超市的寄包柜或快递收件柜,大小包裹由Bundle统一存取,方便又安全。

Bundle内部用于存放数据的实质结构是Map映射,可添加元素、删除元素,还可判断元素是否存在。开发者把Bundle全部打包好只需调用一次putExtras方法,把Bundle全部取出来也只需调用一次getExtras方法。

下面是前一个页面向后一个页面发送请求数据的代码:

     Intent intent = new Intent(MainActivity.this, FirstActivity.class);  // 创建一个目标确定的意图
     Bundle bundle = new Bundle();  // 创建一个新包裹
     bundle.putString("name", "张三");  // 往包裹存入一个字符串
     bundle.putInt("age", 30);  // 往包裹存入一个整型数
     bundle.putDouble("height", 170.0f);  // 往包裹存入一个双精度数
     intent.putExtras(bundle);  // 把快递包裹塞给意图
     startActivity(intent);  // 启动意图所向往的活动页面

下面是后一个页面接收前一个页面请求数据的代码:

     Intent intent = getIntent();  // 获取前一个页面传来的意图
     Bundle bundle = intent.getExtras();  // 卸下意图里的快递包裹
     String name = bundle.getString("name", "");  // 从包裹中取出字符串
     int age = bundle.getInt("age", 0);  // 从包裹中取出整型数
     double height = bundle.getDouble("height", 0.0f);  // 从包裹中取出双精度数

3.5.4 向上一个Activity返回参数

如同一般的通信一样,Intent有时只把请求数据发送到下一个页面就行,有时还要处理下一个页面的应答数据(通常发生在下一个页面返回到上一个页面时)。如果只把请求数据发送到下一个页面,前一个页面调用startActivity方法就可以;如果还要处理一下个页面的应答数据,此时就得分多步处理,详细步骤如下:

步骤01 前一个页面打包好请求数据,调用方法startActivityForResult(Intent intent, int requestCode),表示需要处理结果数据,第二个参数表示请求编号,用于标识每次请求的唯一性。

步骤02 后一个页面接收请求数据,进行相应处理。

步骤03 后一个页面在返回前一个页面时,打包应答数据并调用setResult方法返回信息。setResult的第一个参数表示应答代码(成功还是失败),代码示例如下:

     Intent intent = new Intent();  // 创建一个新意图
     Bundle bundle = new Bundle();  // 创建一个新包裹
     bundle.putString("job", "码农");  // 往包裹存入一个字符串
     intent.putExtras(bundle);  // 把快递包裹塞给意图
     setResult(Activity.RESULT_OK, intent);  // 携带意图返回前一个页面
     finish();  // 关闭当前页面

步骤04 前一个页面重写方法onActivityResult,该方法的输入参数包含请求编号和应答代码,请求编号用于判断对应哪次请求,应答代码用于判断后一个页面是否处理成功。然后对应答数据进行解包处理,代码示例如下:

下面是完整的请求页面代码与应答页面代码,结合效果界面加深对Activity处理参数传递的理解。请求页面的代码示例如下:

应答页面的代码示例如下:

具体的效果图分别如图3-23、图3-24、图3-25所示。其中,图3-23是当前页面要向下一个页面发送请求时的界面,图3-24是下一个页面准备返回上一个页面时的界面,图3-25是上一个页面收到下一个页面应答时的界面。

图3-23 准备向下一个页面发送请求

图3-24 下一个页面准备返回消息

图3-25 上一个页面收到返回消息