Activity 作为一个老生常谈的话题,它是我们刚接触Android开发就遇到,虽然已有一段时间开发经验,但谈起完全搞懂Activity相关,不敢妄言,故结合个人理解及书籍参考,简单总结一下 Activity基础相关知识,其中也留出了一些有意思的问题。
大体的脑图如下(启动流程涉及的过多,暂时放在后期):
基础的生命周期方法这里就不做解释了,大家刚开始接触时就是这些方法了。
实际使用来说,他们看起来的确差不多,但是 onStart和onStop 是从Activity是否可见这个角度来回调的,而 onResume 和 onPause是从Actvity是否位于前台这个角度来回调的,除了这两点,实际使用中并无其他区别。
A的 onPause 先执行。
在Android的官方文档中,在旧的Activity onPause执行完之后,新的Activity 才能onResume,所以我们应该尽量避免在 onPause 中做太多耗时操作,尽量应该放到onStop中。
在我们开发中,经常会遇到转屏的问题,而转屏一般也会带来 Activity的重新创建,所以大多数开发者开发的时候,Activity默认是禁止转屏的,但是在一些短视频软件上,转屏就是一件非常常见的事了,那么如何处理相应数据的保存与恢复就是我们必须关注的事了。
当系统配置发生改变后,Activity会被销毁,其 onPause,onStop,onDestory均会被调用,同时由于Activity是在异常情况下终止的,系统会调用 onSaveInstanceState 来保存当前 Activity 的状态。这个方法的调用时机是在 onStop 之前,它和onPause 没有既定的时序关系,有可能在onPause之前调用,也有可能在 onPause之后调用。但需要注意的是,这个方法只会出现在 Activity 被异常终止的情况下。正常情况下不会回调这个方法。
当Actiivty 被重新创建后,系统会调用 onRestoreInstanceState, 并且吧 Activity 销毁时 onSaveInstanceState 方法保存的 Bundle 对象作为参数同时传递给 onRestoreInstanceSate 和 onCreate 方法。因此我们可以通过 onRestoreInstanceState和 onCreate 方法来判断 Activity是否被重建了,如果被重建了,那么我们就可以去除之前保存的数据并恢复,从时序上来说,onRestoreInstanceState 的调用时机在 onStart之后。
在上面我们知道,当Activity因为异常情况发生重建时,系统会主动调用 onSaveInstanceState 方法来进行保存,但需要注意的是 onSaveInstanceState 并不适合于保存大量数据,Google的推荐是用其来保存相应的id及key,而相应的大量数据推荐使用ViewModel进行保存。
onSaveInstanceState 是为了保存应用进程在后台时候由于内存限制而被终止,或者配置更改时的回调,其 默认实现是保存了关于 activity的视图层次状态的临时信息,比如EditText中输入的文本,Rv,Lv的滑动位置等,其支持的类型只是Bundle,所以并不适合存储大量数据,适合于少量的临时数据。
ViewModel 可以代理复杂数据的加载,也可以作为临时的存储位置,但是不能在手动 finish 的进程中存留,它的意义更多的是实现 当系统状态更改时,实现数据的保留,而不是ui状态的保留。
在默认情况下,当我们多次启动同一个Activity 时,系统会创建多个实例并把他们按照 后进先出的原则(栈结构) 一一放进任务栈中。任务栈是一种 后进先出 的栈结构。每次返回时都会销毁一个 Activity 。所以为了更好的管理Activity 和对Activity进行利用,Android提供了 启动模式来解决这个问题,分别为standard,SingleTop,singleTask,SingleInstance
标准模式,也是默认模式。即每次启动一个Activity 都会重新创建一个新的实例,不论这个实例是否存在。相应的生命周期也遵从标准的生命周期过程。
栈顶复用模式。简单理解为,如果新的Activity采用这个模式启动,如果此Activity已经处于当前任务栈栈顶,那么此Activity不会被重复创建,当调用 startActivity跳转时,会回调它的 onNewIntent() 方法。通过此方法我们可以取出当前请求的信息。需要注意的是如果使用 startActivityForResult 跳转,将忽略启动模式。
栈內复用模式。这是一种单实例模式,在这种模式下,只要Activity在一个栈中存在,那么多次启动此Activity都不会创建实例,和 singleTop 一样,系统也会回调 onNewIntent. 简单理解为,当启动一个 启动模式为 singleTask的Activity时,系统会再栈里寻找是否存在此Activity,如果找到,将此Activity顶部的所有Activity全部出栈,并把其调到栈顶并调用它的 onNewIntent 方法。如果不存在此Activity,则创建此Activity并压入栈顶。
单实例模式,又称加强的 singTask 模式。它除了具有singleTask的所有特性外,还具有额外的特性,那就是 具有此启动模式的Activity 只能单独位于一个 任务栈中。
Activity的Flags有很多,这些标记位在我们实际开发中帮助很大,其中有些标记位可以设定 Activity的启动模式,比如 使用 Application 启动Activity时 添加的 FLAG_ACTIVITY_NEW_TASK 等。
为Activity指定 singleTask 的启动模式. 问题:为什么Application调用startActivity需要附带此tag?
为Activity指定 singleTop 的启动模式
销毁目标 Actiivty 和它之上的所有Activity并重新创建。如果搭配 FLAG_ACTIVITY_SINGLE_TOP 使用,会销毁它之上的所有Activity,如果目标实例存在,则不会销毁,会调用其的 onNewIntent 方法。
具有这个标记的 Activity不会出现在历史Activity列表,等同于 xml中指定Activity的属性 android:excludeFromRecents=“true”。
注意,这个参数如果放在首个Activity,那么接下来这个Activity栈的所有Activity都会受到影响。
启动一个Activity分为两种,显式和隐式,日常开发中我们用的最多的就是显式,而隐式常常用作,h5打开一个app,或者调用系统某个设置页,打开相机等等。隐式调用相比显式调用来说,稍微复杂一点,它需要Intent能够匹配目标组件IntentFilter
中所设置的过滤信息,如果不匹配将无法启动目标Activity。 IntentFilter 中的过滤信息 有 action,category,data
action是一个字符串,系统预定义了一些action,同时我们也可以定义我们自己的action。action的匹配规则是Intent中的action必须能够和过滤规则中的action匹配。而过滤规则我们可以定义多个,只要Intent中的action能与任意一个过滤规则匹配就是匹配成功。如果你的Intent中没有定义action,则匹配失败。
<action android:name="android.intent.action.VIEW" />
常用的action如下:
category是一个字符串,系统也为我们预制了一席,对于在 已经定义的匹配规则,在Intent 中存在的categoty必须全部符合已经定义了的规则,当然也可以不填,如果Intent中没有包含,系统会为我们默认带上 android.intent.category.DEFAULT,这一点和action 有一些区别.
<category android:name="android.intent.category.DEFAULT" />
data的匹配规则和 action 类似,如果定义了相应的匹配规则,那么Intent中必须包含相应的data.
<data
android:host="*"
android:mimeType="string"
android:path="/test"
android:pathPattern="*"
android:pathPrefix="/petterp"
android:port="1080"
android:scheme="http" />
data由两部分组成,mimeType和URI,mimeType指媒体类型,比如image/jpeg. Audio/mpeg4-genric和 video/*等,可以表示图书,文本,视频等不同的媒体形式。而URI中包含的数据就相对多一点。
://:/[||]
data的匹配规则和action 类似,它也要求 Intent 中必须含有 data数据,并且 data数据能够完全匹配过滤规则中的某一个 data,这里的完全匹配是指 过滤规则中出现的 data部分也出现在了 Intent 中的 data中。
如下所示匹配规则:
<intent-filter>
<action android:name="petterp.test1" />
<data android:mimeType="image/*" />
intent-filter>
这种匹配规则指定了媒体类型为所有类型的图片,那么Intent中的 mineType 属性必须为 image/* 才能匹配,这种情况下虽然过滤规则没有指定URI,但是却有默认值。URI 的默认值为 content 和 file 。也就是说,虽然没有指定 URI,但是Intent中的 URI 部分的 schema 必须为 content 或者 file才能匹配。
所以如果我们要调用的话,可以写出如下的代码:
val intent=Intent("petterp.test1")
intent.setDataAndType("file://abc","iamge/*")
接着,我们再看一个案例,相比上面的,复杂一点
<intent-filter>
<action android:name="petterp.test2" />
<category android:name="android.intent.category.DEFAULT" />
<data
android:host="www.baidu.com"
android:mimeType="image/*"
android:path="/test"
android:port="8080"
android:scheme="https"
android:pathPattern=".*"/>
intent-filter>
对于这个案例,我们在调用的时候,就必须采用以下格式:
val intent = Intent("petterp.test2")
intent.setDataAndType(Uri.parse("https://www.baidu.com:8080/test/路径随便"),"image/*")
如果这里看不懂的话,多看一遍data匹配规则就行。
注意:category必须加,就算Intent(intent中默认会带一个)中不加,xml中也必须带,否则会存在找不到相应的Activity,原因是因为不含有 DEFAULT 这个 category的Action是无法接收Intent的。
非常高兴你能看到这里,虽然这只是Activity的一些基础,但都是我们每个开发者最基本应该掌握的。勉励前行
参阅资料