《Android 艺术探索》第 1 章 Activity 的生命周期和启动模式

一、Activity 生命周期

  1. 当前 Activity A 打开一个新的 Activity B,生命周期调用顺序是怎样的?
    — 顺序为:A.onPause() -> B.onCreate() -> B.onStart() -> B.onResume -> A.onStop() -> A.onDestroy()
    注意点:
  • 不能在 onPause() 中执行重量级操作,因为必须当前 Activity 的 onPause() 执行完成后才能 Resume 新的 Activity.
  • 若 B Activity 是透明主题,则 A 不会执行 onStop() 方法,因此此时 A 还是可见的。(待验证 -> YES)
  1. onStart() 和 onResume()、onPause() 和 onStop() 从描述上差不多,但是有什么实质性的不同呢?
    — onStart() 和 onStop() 是从 Activity 是否可见这个角度来回调的;onResume() 和 onPause() 是从 Activity 是否位于前台这个角度来回调的,除此区别,在实际使用种无甚区别。

  2. onSaveInstanceState() 方法仅在 Activity 被异常杀掉才会调用吗?onRestoreInstanceState() 在什么时候会被调用?参数 Bundle 有什么作用?Activity 异常被杀死重新恢复后,页面上有些控件的状态会丢失,有些会保留,怎么处理的?
    — TODO 待验证 。

二、Activity 启动模式

  1. 如何理解 “任务栈” 这个概念?
  • 系统启动同一个 Activity 时会创建多个实例并将其放入任务栈中,当单击 back 键时,这些 Activity 会一一回退。
  • 任务栈 是一种 “后进先出” 的栈结构。当栈中没有任何 Activity 时,系统会回收这个任务栈。
  • 默认情况下,所有 Activity 所需的任务栈的名字都是 应用的包名。
  • 若为 Activity 单独指定任务栈名,则可指定 taskAffinity 属性。需要注意的是该属性主要和 singleTask 启动模式配对使用,其他场景无用。
  1. 四种启动模式含义及使用场景。
  • standard:标准模式,也是系统默认模式,每次启动都会重新创建一个新的实例,不管这个实例是否已经存在。 启动 standard 模式的 Activity ,该 Activity 会位于 启动它的页面的 任务栈中。
  • singleTop:栈顶复用模式。如果新的 Activity 已经位于栈顶,则此 Activity 不会被重新创建,同时它的 onNewIntent() 方法也会被回调;若新的 Activity 没有位于栈顶,则仍然会重新创建一个新的实例。
  • singleTask:栈内复用模式。一种单例模式,只要 Activity 在一个栈中存在,则多次启动该 Activity 都不会重新创建实例,此时它的 onNewIntent() 方法也会被回调;若没有任何栈中存在该 Activity,则需要先创建该 Activity 所在的任务栈 S2,再创建该 Activity 实例,并入栈到 S2。【问题:若 singleTask 启动的 Activity 没有自定义任务栈,是否为其创建的任务栈也为 应用名称?】
  • singleInstance:单实例模式。具有此种模式的 Activity 只能单独地位于一个任务战中,也即是,Activity A 以 singleInstance 模式启动,则系统会为 A 创建一个新的任务栈,然后这个任务栈中只能存有 A 一个实例。

注:

  • singleTop、singleTask 模式,若非重新创建 Activity,则启动时都会只回调 onNewIntent() 方法,所有,注意此种模式下,需要把 onCreate() 中的部分功能按逻辑复制一份到 onNewIntent() 中。
  • singleTask 带有 clearTop 的作用,也即是若栈中已经存在 Activity 实例,则会将该 Activity 之上的其他所有 Activity 都清除掉。
  • singleTask 的三种操作场景,需要先检测 Act 的任务栈 S2 是否存在,再检查该任务栈中是否存在该 Act 实例,主要表现为:1)任务栈 S2 不存在,则先创建一个任务栈,再创建该 Act 实例并入栈;2)任务栈 S2 存在,若S2中不存在该 Act 实例,则创建该 Act 实例并入栈;3)若任务栈中存在该 Act 实例,则会先清除栈中位于该 Act 之上的其它所有 Act,然后将该 Act 切到栈顶。
  • 查看应用当前任务栈情况:adb shell dumpsys activity

问题:

  • 若 A 为 standard,所在任务栈为 taskA,B 为 singleTask,则 A 启动 B,在 A 的 taskA 中没有找到 B 的实例,则系统会为 B 新建一个 taskB,然后新建 B 的实例,并入栈到 taskB 中。若没有为 B 指定任务栈名称/ taskAffinity 属性,那么 taskB 的名称是什么?【TODO 待验证】
  1. 前台任务栈 和 后台任务栈的区别。
    — 前台任务栈就是当前展示的 Activity 所在的任务栈;后台任务栈就是位于后台的 Activity 所在的任务栈。

  2. Activity 的 Flags 用法。

  • FLAG_ACTIVITY_NEW_TASK:同 “singleTask”。
  • FLAG_ACTIVITY_SINGLE_TOP:同 "singleTop" 。
  • FLAG_ACTIVITY_CLEAR_TOP:当 Activity 启动是,在同一个任务栈中所有位于其之上的 Activity 都会被清理。一般需要和 FLAG_ACTIVITY_NEW_TASK 配合使用。此种情况下,若被启动 Activity 实例存在,则其之上 Act 都会被清除;若启动 Activity 是 standard,也即是在 Intent 中添加此 FLAG 启动 standard 的 Act,则连同这个 Act 本身及其之上的都会被清除,系统会重新创建新的 Activity 并入栈
  • FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS:具有该标记的 Activity 不会出现在历史 Activity 的列表中。

三、IntentFilter 的匹配规则

  1. 隐式和显式启动 Activity 。
  • 显式启动:明确该启动对象的组件信息,包括包名和类名。
  • 隐式启动:Intent 能够匹配 目标组件的 IntentFilter 中设置的过滤信息。因素有:action、category、data .
  1. 如何通过隐式调用启动一个 Activity ?
  • 需要同时匹配 action、category、data 信息,否则匹配失败;
  • 若没有指定 category,则均默认为 "android.intent.category.DEFAULT" ;
  • 一个 Activity 中可以有多个 intent-filter,一个 Intent 只要能匹配任何一组 intent-filter 即可成功启动对应的 Activity 。
  1. 如何通过 scheme 启动另外一个应用 / 另外一个应用的某个页面?
    方式一:通过 Intent 指定组件 包名和 类名即可。
Intent intent = new Intent(Intent.ACTION_MAIN); 
intent.addCategory(Intent.CATEGORY_LAUNCHER);             
ComponentName cn = new ComponentName(packageName, className);             
intent.setComponent(cn); 
startActivity(intent);

// 打开某个页面,则可直接指定 action 并且需要翻开对外访问权限:exported=true 。

方式二:通过 data 配置 scheme uri 进行跳转并可传递参数。

// 应用 A 启动应用 B
Uri uri = Uri.parse("bocmcht://payresult/mobile?jsonData=123");
Intent intent = new Intent(Intent.ACTION_VIEW, uri);
startActivity(intent);

// 应用 B 清单文件声明
<intent-filter>
         <action android:name="android.intent.action.VIEW"/>
         <category android:name="android.intent.category.DEFAULT"/>
         <data android:scheme="bocmcht" android:host="payresult" android:path="/mobile"/>
</intent-filter>

// 应用 B 获取应用 A 传递的参数
Uri uri = getIntent().getData();
String scheme = uri.getScheme();// 打印:bocmcht
String host = uri.getHost();//打印:payresult
String path = uri.getPath();//打印:/mobile
String queryString = uri.getQuery();//打印:jsonData=123
  1. H5 如何通过 scheme 打开一个 Native 页面?
    — 同上。action 需要设置为 "android.intent.action.VIEW" 才行,并且要配置 category,如下:
<!--scheme 允许在浏览器中打开-->
<category android:name="android.intent.category.BROWSABLE"/>
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,692评论 6 501
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,482评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,995评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,223评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,245评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,208评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,091评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,929评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,346评论 1 311
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,570评论 2 333
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,739评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,437评论 5 344
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,037评论 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,677评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,833评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,760评论 2 369
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,647评论 2 354