我们知道Android 有四大组件,其中使用最频繁的就是 Activity,可以说应用的很多界面都是Activty,所有对于Android的基础来说,我们要好好认识一下Activity.
Activty的生命周期
-
先来看看谷歌官方Android开发者文档一张关于Activity生命周期的截图
-
在应用正常运行的情况下,Activity会经历上图中的七个生命周期方法,
- onCreate():Activity第一次启动的时候调用,也是生命周期的第一个方法,我们可以在其做一些Activity数据初始化的工作,还有最常用的使用setContentView方法加载布局。
- onStart():该方法调用的时候,Activity已经显示出来了,正在被启动,但是我们还看不到(Activity还在后台),不能与用户进行交互,无法获取焦点。
- onResume():该方法被调用的时候,Activity已经完全显示出来(显示在前台),我们用户可以与进行交互。
- onPause():和onStart()对应,该方法调用时Activity正在被停止,Activity已经失去焦点,该方法可以做一些不太耗时的操作,比如存储数据,停止动画等,太耗时的重量级操作不能在该方法中执行
- onStop():调用了onPause()方法紧接着就是调用该方法,表示Activity即将停止,与onStart()对应,Activity此时已经不可见,在该方法中也已做一些回收的操作。
- onRestart():该方法表示Activity重新启动,当用户点击Home键,当前Activity变为不可见状态( onPause()和onStop()会被调用),用户重新进入应用,就会调用onRestart()方法,也就是Activity从不可见状态又变成可见状态
- onDestory():该方法被调用的时候表示Activity马上就好销毁了,这时我们可以在这个回调方法中做一些回收操作和资源释放(比如解绑广播)。
异常情况下的生命周期分析
onSaveinstancestate()保存Activity被销毁的状态,onRestoreoninstancestate()方法在Activity被重建之后该Bundle对象保存销毁前的Activity的状态(比如TextView 可以保存文本选择状态和选中内容),正常情况下这两个方法是不会被回调的,系统只有在Activity异常终止的时候才会调用者两个方法。
Activty的启动模式
我们知道,Activity的维护是在任务栈中,而栈的数据结构是先进后出,在默认的情况下,只要新创建一个Activity,就把Activity放入任务栈中,我们可以这样理解,默认情况下打开一个Activity叫做进栈,关闭一个Activity叫出栈,我们每次操作Activity都是任务栈栈顶的Activity
-
为什么会有启动模式
- 在Activity的默认启动模式下,当我们多次启动同一个Activity,则系统重复创建这个Activity实例放入任务栈中,这样是不合理的,所以Android系统设计的时候也考虑到了这个问题,提供了不同的启动模式来修改Activity的启动方式,他们分别是standard(默认模式)、singleTop(栈顶复用模式)、singleTask(栈内复用模式)、singleInstance(单实例模式)
standard(模式模式):在没有给Activity指定启动模式时候的系统默认模式,每次启动一个Activity都会重新创建一个Activity新的实例,放入任务栈的栈顶,被创建的Activity都会调用onCreate(),onStart(),onResume()三个方法。一个任务栈可以有多个实例,而每个实例也可以属于不同的任务栈,在默认模式下,哪个Activity启动了另一个Activity,则这个Activity就会运行在启动它的Activity的任务栈中(当使用Application启动Activity的时候,是会报错的,因为Application类型的context没有任务栈,这是需要加入一个FLAG_ACTIVITY_NEW_TASK标记,创建一个任务栈,这时启动的Activity实际是以singleTask模式启动的)。
singleTop(栈顶复用模式):在该模式下,如果启动的Activity已经位于任务栈的栈顶,那么此Activity不会被重新创建,并且该Activity的onNewIntent方法回被调用,通过该方法的参数我们得到当前请求的信息,而onCreate(),onStart()不会被回调,因为Activity并没有改变(例如浏览器的书签Activity就可以应用该模式)
singleTask(栈内复用模式):也可以叫栈内单一实例模式,在该模式下,启动一个Activity会先检查栈内是否有该Activity的实例,有该实例则不会重复创建,并且把在该实例前面的Activity全部出栈,也同样会调用onNewIntent方法,如果不存在则会新创建一个任务栈,并把该Activity放入任务栈中。
singleInstance(单实例模式):可以说这个模式是singleTask模式的加强版,在该模式下Activity会单独占用一个任务栈,在这个任务栈中由于栈内复用性,后续请求不会创建一个新的Activity实例,除非该任务栈被系统销毁。(该模式可以应用于我们手机接电话的界面,不管有多少个电话打进来,只能有一个接电话的界面,如果是不是单实例模式,如果有多个电话打进来,手机就可能由于多个Activity占用内存太多而卡死了)。
IntentFilter匹配规则
- 相信对于Activity启动方式大家都很熟悉,主要有显示调用和隐式调用;一个Actiicty的清单配置可以有多个IntentFilter,一个Intent只要能匹配任何一组IntentFilter,既可以启动对应的Activity
- 显示调用
- 显示调用很简单,明确指定要启动Activity的包名和类名
- 显示调用
Intent intent = new Intent(WelcomeActivity.this,MainActivity.class);
- 隐式调用
- 隐式调用则要根据要启动Activity在清单文件(AndroidManifest.xml)中的IntentFilter所设置的过滤信息进行设置匹配,才能启动这个Activity,来看一个例子
<activity android:name=".TestActivity">
<intent-filter >
<action android:name="com.mao.action.Test"/>
<action android:name="com.mao.action.Test1"/>
<category android:name="com.mao.category.Test"/>
<category android:name="com.mao.category.Test2"/>
<category android:name="android.intent.category.LAUNCHER" />
<data android:mimeType="image/*"
android:host="mao"/>
</intent-filter>
</activity>
-
从例子可以看到,过滤信息中三种过滤信息, action、category、data;当要使用隐式调用方式启动Activity且该Activity包含这个三种过滤信息,必须同时匹配这些过滤信息,才能启动该Activity;每一种过滤信息代表一个类别,每一个类别可以有多个,同一类别的过滤信息共同约束当前类别的匹配,也就是一个Intent必须同时匹配action、category、data三个类别才能匹配成功开启Activity,下面逐一对各个类别匹配规则进行分析
-
action : 它是一个字符串,Android系统给我们预定义了一些action,我们也可以自己定义action,比如上面例子中的“com.mao.action.Test”。
- action的匹配必须是字符串一模一样(大小写也要一样),不管action设置了多少个,只要Intent设置了其中一个就算匹配成功action类别,否则匹配失败。
categroy :这个类别也是一个字符串,Android系统给我们预定义了一些categroy,和action不同的是,action是Intent必须有的,而categroy是Intent可有,也可以没有,但是一旦有,Intent调用addCategroy()设置的字符串必须和定义的categroy中的任何一个相同。为什么可以没有呢?在系统调用 startActivity()或者startActivityForResult()方法时,会自动加上android.intent.category.LAUNCHER 这个categroy,所以Intent不设置categroy这个类别也可以,而我们要隐式启动,则在intent-filter中就必须加上<category android:name="android.intent.category.LAUNCHER" />这个配置。
-
data : data类别的匹配规则和action类似,如果设置这个匹配规则,Intent中就必须调用setDataAndType()方法设置data,data由mimeType(媒体类型)和URI两部分组成,
- data的语法和匹配规则
-
<data
android:scheme="string" //URI的模式,比如http、file、content,没有指定,则URI无效
android:mimeType="string"//媒体类型,例如image/jpeg、audio/mpeg4-generic、video/*
android:host="string"//URI的主机名,没有指定则URI无效
android:path="string"//URI端口号
android:port="string"//完整路径信息
android:pathPattern="string"//完整路径信息
android:pathPrefix="string"//路径前缀信息
/>
例子(1):
<data android:mimeType="image/*"
android:host="mao"/> 这里没有设置scheme,因为scheme有默认值content和file
intent.setDataAndType(Uri.parse("file://mao"),"image/png");//必须调用该方法,不能分别调用setData()和setType方法,这两个方法会清除对方的值
例子(2):
<data android:scheme="http"
android:mimeType="video/mpeg"
android:host="www.mao.cn"
/>
intent.setDataAndType(Uri.parse("http://www.mao.cn"),"image/mpeg");
相信到了这里你已经懂得如何隐式启动一个Activity了,前面例子中的Activity我们可以这样隐式启动
Intent intent=new Intent("com.mao.action.Test");
intent.addCategory("com.mao.category.Test");
intent.setDataAndType(Uri.parse("file//mao"),"image/mpeg");
startActivity(intent);
最后
知识久了不用就会忘记,我希望自己看到这一篇文章的时候能将遗忘的一些基础知识重新找回,也希望能帮助到看到这篇文章的你。文章中如果有错误,请大家给我提出来,大家一起学习进步,如果觉得我的文章给予你帮助,也请给我一个喜欢或者关注。
- 参考资料
- 《Android开发艺术探索》