要点提炼|开发艺术之 Activity

作者: 厘米姑娘 | 来源:发表于2017-12-06 23:13 被阅读218次

本系列专题以提炼要点的方式总结知识点,而不做具体展开,对应的会附上个人较喜欢的文章链接供大家详细了解和学习。第一个板块“开发艺术”是对任老师的《Android开发艺术探索》著作的学习和扩展,此书的目录也是该板块的主线,那么就先从我们熟悉的Activity开始吧~

在之前的Activity篇主要学习Activity的生命周期、创建和页面跳转的实现,本篇将深入了解Activity,学习清单:

  • 生命周期全解析
  • 四种启动模式
  • IntentFilter匹配规则

一.生命周期全解析

1.Q:典型情况下Activity生命周期
A:
a.onCreate():

  • 状态:Activity 正在创建
  • 任务:做初始化工作,如setViewContent界面资源、初始化数据
  • 注意:此方法的传参Bundle为该Activity上次被异常情况销毁时保存的状态信息。

b.onStart():

  • 状态:Activity 正在启动,这时Activity 可见但不在前台,无法和用户交互。

c.onResume():

  • 状态:Activity 获得焦点,此时Activity 可见且在前台并开始活动。

d.onPause():

  • 状态: Activity 正在停止
  • 任务:可做 数据存储、停止动画等操作。
  • 注意:Activity切换时,旧Activity的onPause会先执行,然后才会启动新的Activity。

e.onStop():

  • 状态:Activity 即将停止
  • 任务:可做稍微重量级回收工作,如取消网络连接、注销广播接收器等。
  • 注意:新Activity是透明主题时,旧Activity都不会走onStop。

f.onDestroy():

  • 状态:Activity 即将销毁
  • 任务:做回收工作、资源释放

g.onRestart():

  • 状态:Activity 重新启动,Activity由后台切换到前台,由不可见到可见。

onStart()和onResume()、onPause()和onStop()的区别: onStart与onStop是从Activity是否可见这个角度调用的,onResume和onPause是从Activity是否显示在前台这个角度来回调的,在实际使用没其他明显区别。

推荐阅读:对Activity生命周期方法的感性理解

2.Activity生命周期的切换过程
①启动一个Activity:

  • onCreate()-->onStart()-->onResume()

②打开一个新Activity:

  • 旧Activity的onPause() -->新Activity的onCreate()-->onStart()-->onResume()-->旧Activity的onStop()

③返回到旧Activity:

  • 新Activity的onPause()-->旧Activity的onRestart()-->onStart()-->onResume()-->新Activity的onStop()-->onDestory();

④Activity1上弹出对话框Activity2:

  • Activity1的onPause()-->Activity2的onCreate()-->onStart()-->onResume()

⑤关闭屏幕/按Home键:

  • Activity2的onPause()-->onStop()-->Activity1的onStop()

⑥点亮屏幕/回到前台:

  • Activity2的onRestart()-->onStart()-->Activity1的onRestart()-->onStart()-->Activity2的onResume()

⑦关闭对话框Activity2:

  • Activity2的onPause()-->Activity1的onResume()-->Activity2的onStop()-->onDestroy()

⑧销毁Activity1:

  • onPause()-->onStop()-->onDestroy()

参考文章:实际体验Activity生命周期

3.Q:生命周期的各阶段
A:
a.完整生命周期
Activity在onCreate()和onDestroy()之间所经历的。
在onCreate()中完成各初始化操作,在onDestroy()中释放资源。

b.可见生命周期
Activity在onStart()和onStop()之间所经历的。
活动对于用户是可见的,但仍无法与用户进行交互。

c.前台生命周期
Activity在onResume()和onPause()之间所经历的。
活动可见,且可交互。

4.onSaveInstanceState和onRestoreInstanceState
a.出现时机:异常 情况下Activity 重建,非用户主动去销毁

b.系统异常终止时,调用onSavaInstanceState来保存状态。该方法调用在onStop之前,但和onPause没有时序关系。

onSaveInstanceState与onPause的区别:前者适用于对临时性状态的保存,而后者适用于对数据的持久化保存。

c.Activity被重新创建时,调用onRestoreInstanceState(该方法在onStart之后),并将onSavaInstanceState保存的Bundle对象作为参数传到onRestoreInstanceState与onCreate方法。

可通过onRestoreInstanceState(Bundle savedInstanceState)和onCreate((Bundle savedInstanceState)来判断Activity是否被重建,并取出数据进行恢复。但需要注意的是,在onCreate取出数据时一定要先判断savedInstanceState是否为空。另外,谷歌更推荐使用onRestoreInstanceState进行数据恢复。

推荐阅读onSaveInstanceState和onRestoreInstanceState详解

5.Activity异常情况下生命周期分析

a.由于资源相关配置发生改变,导致Activity被杀死和重新创建
例如屏幕发生旋转:当竖屏切换到横屏时,会先调用onSaveInstanceState来保存切换时的数据,接着销毁当前的Activity,然后重新创建一个Activity,再调用onRestoreInstanceState恢复数据。

  • onSaveInstanceState-->onPause(不定)-->onStop-->
    onDestroy-->onCreate-->onStart-->onRestoreInstanceState-->onResume

为了避免由于配置改变导致Activity重建,可在AndroidManifest.xml中对应的Activity中设置android:configChanges="orientation|keyboardHidden|screenSize"。此时再次旋转屏幕时,该Activity不会被系统杀死和重建,只会调用onConfigurationChanged。因此,当配置程序需要响应配置改变,指定configChanges属性,重写onConfigurationChanged方法即可。

b.由于系统资源不足,导致优先级低的Activity被回收
①Activity优先级排序:

  • 前台可见Activity>前台可见不可交互Activity(前台Activity弹出Dialog)>后台Activity(用户按下Home键、切换到其他应用)

②当系统内存不足时,会按照Activity优先级从低到高去杀死目标Activity所在的进程。

③若一个进程没有四大组件在执行,那么这个进程将很快被系统杀死。

推荐阅读异常情况下的生命周期分析


二.Activity四种启动模式

1.Q:设置Activity启动模式的方法
A:
a.在AndroidManifest.xml中给对应的Activity设定属性android:launchMode="standard|singleInstance|single Task|singleTop"。
b.通过标记位设定,方法是intent.addFlags(Intent.xxx)

2.Q:Activity的四种LaunchMode
A:
a.standard:标准模式、默认模式

  • 含义:每次启动一个Activity就会创建一个新的实例。
  • 注意:使用ApplicationContext去启动standard模式Activity就会报错。因为standard模式的Activity会默认进入启动它所属的任务栈,但是由于非Activity的Context没有所谓的任务栈。

b.singleTop:栈顶复用模式

  • 含义:如果新Activity已经位于任务栈的栈顶,就不会重新创建,并回调onNewIntent(intent)方法。

c.singleTask:栈内复用模式

  • 含义:只要该Activity在一个任务栈中存在,都不会重新创建,并回调onNewIntent(intent)方法。如果不存在,系统会先寻找是否存在需要的栈,如果不存在该栈,就创建一个任务栈,并把该Activity放进去;如果存在,就会创建到已经存在的栈中。

d.singleInstance:单实例模式

  • 含义: 具有此模式的Activity只能单独位于一个任务栈中,且此任务栈中只有唯一一个实例。

标识Activity任务栈名称的属性:android:taskAffinity,默认为应用包名。

推荐阅读:Activity的四种LaunchMode详解

3.常用的可设定Activity启动模式的标记位

FLAG_ACTIVITY_SINGLE_TOP:对应singleTop启动模式。
FLAG_ACTIVITY_NEW_TASK :对应singleTask模式。


三.IntentFilter匹配规则

原则:
①一个intent只有同时匹配某个Activity的intent-filter中的action、category、data才算完全匹配,才能启动该Activity。
② 一个Activity可以有多个 intent-filter,一个 intent只要成功匹配任意一组 intent-filter,就可以启动该Activity。

a. action匹配规则:

  • 要求intent中的action 存在且必须和intent-filter中的其中一个 action相同。
  • 区分大小写。

b. category匹配规则:

  • intent中的category可以不存在,这是因为此时系统给该Activity 默认加上了< category android:name="android.intent.category.DEAFAULT" />属性值。
  • 除上述情况外,有其他category,则要求intent中的category和intent-filter中的所有category 相同。

c. data匹配规则:

  • 如果intent-filter中有定义data,那么Intent中也必须也要定义date。
  • data主要由mimeType(媒体类型)和URI组成。在匹配时通过intent.setDataAndType(Uri data, String type)方法对date进行设置。
  • 要求和action相似:如果没有指定URI,默认值为content和file; 有多组data规则时,匹配其中一组即可。

采用隐式方式启动Activity时,可以用PackageManager的resolveActivity方法或者Intent的resolveActivity方法判断是否有Activity匹配该隐式Intent。

推荐阅读:Intent和IntentFilter


希望这篇文章对你有帮助~

相关文章

网友评论

  • hai_phon:我喜欢你 不 我不配
    厘米姑娘:@hai_phon (`・ω・´)ノ戏精
  • 爱吃板栗的小女孩:大神~是setContentView 不是setViewContent啦
    厘米姑娘:@爱吃板栗的小女孩 🌚
  • 三次元蚂蚁:楼主总结的很棒,但是关于onSaveInstanceState的原理实现上,我一直不是很理解,希望楼主帮忙解答一下我的疑问,onSaveInstanceState把数据存在了内存中,当横竖屏切换时会调用onDestroy回收内存,为何还能从onRestoreInstanceState中恢复内存中的数据?还有就是如果因为内存不足导致activity所在的进程被杀死,会像onDestroy方法一样回收内存吗,如果会的话,又是怎么从内存中恢复数据的?
    厘米姑娘:@StrongWind 😂猜测的猜测的
    三次元蚂蚁:@minmin_1123 谢谢小姐姐,你真的很厉害!
    厘米姑娘:1. 我理解的onDestory只是表明Activity的生命周期结束了,而像资源啊等这些对象所占据的内存是靠GC去回收的,那么具体回收策略跟虚拟机有关系,因此我猜测这部分数据被暂时保留还没被回收
    2.onSaveInstanceState和onRestoreInstanceState的调用是需要时机的,比如内存不足,所以都会先把原Activity销毁掉然后再重建
  • MrAllTheSunday:总结的很棒。很佩服楼主的专业性。有一点,想探讨一下 onSaveInstanceState 在activity的调用时机,我想 当一个activity变得“容易”后台杀掉(例如,按右键切换应用,按home键退后台)都会调这个方法。异常并不一定出现。
    MrAllTheSunday:总结的很赞:+1:
    厘米姑娘:@MrAllTheSunday |・ω・`)我也不过是站在巨人肩膀上学习和总结,有问题还是欢迎指正~
    厘米姑娘:@MrAllTheSunday 嗯嗯,你说的这个情况确会有,这么说更严谨点~
  • A挑战未来A:想知道女汉子最后入主哪家大厂了?????
  • dw_0920:FLAG_ACTIVITY_NEW_TASK对应的启动模式不是singleTask吧,在singleTask模式下,若当前栈内存在目标Activity,就把目标Activity上的Activity全部出栈,并展示目标Activity。 这种实现和FLAG_ACTIVITY_CLEAR_TOP类似
    dw_0920:@minmin_1123 FLAG_ACTIVITY_NEW_TASK只是开启新的Activity栈吧
    厘米姑娘:欢迎补充~
    厘米姑娘:你说的那个FLAG_ACTIVITY_CLEAR_TOP确实和singleTask实现类似,但是FLAG_ACTIVITY_NEW_TASK也有这样的效果
  • 招財进寶:是美女吗?能留个微信方便交流技术吗?哈哈
    厘米姑娘:@招財进寶 是女装的哈哈
  • 大清新_6664:总结的 很好啊。
    厘米姑娘:@大清新_6664 (•ө•)♡
  • 逝水流年轻染尘_:请问一下女装大佬,要记住这些理论知识有什么技巧没?:relieved:
    厘米姑娘:重理解,多实践,多复习!

本文标题:要点提炼|开发艺术之 Activity

本文链接:https://www.haomeiwen.com/subject/vudrixtx.html