笔记来源:任玉刚《Android开发艺术探索》
一. Activity生命周期切换
Activity生命周期切换过程二. 正常情况下的生命周期分析
2.1.onStart和onResume,onPause和onStop的区别
onStart和onStop是从Activity 这个角度来回调的。
onResume和onPause是从Activity 这个角度来回调的。
onStart的时候,Activity可见但还在后台(我们看不到),onResume的时候Activity才显示在前台
2.2. 当前Activity为A,此时用户打开一个新ActivityB,那么A的onPause和B的onResume哪个先执行?
必须A的onPause执行完成以后,ActivityB才能启动
三. 异常情况下的生命周期分析
如横竖屏切换,默认情况下,Activity会被销毁并重新创建,非默认情况下,可以阻止系统重新创建Activity
3.1. 用来保存当前Activity的状态,用来恢复。系统才会调用这两个方法,其他情况不会触发该过程
3.2 onSaveInstanceState在onStop之前调用,和onPause无时序关系。onRestoreInstanceState在onStart之后调用
3.3 当AndroidMainfest.xml文件给Activity设置android:configChanges="orientation|screenSize|keyboardHidden"时,Activity异常中止时不会重建Activity,onSaveInstanceState和onRestoreInstanceState不会被调用,取而代之的是调用了onConfigurationChanged方法
四. Activity的启动模式
4.1
每次启动一个Activity都会重新创建一个新的实例,不管这个实例是否已经存在。谁启动新Activity,新Activity就运行在启动它的那个Activity的栈中
4.2
如果新Activity已经位于任务栈的栈顶,那么此Activity不会被重建,onCreate,onStart不会被调用,其onNewIntent方法会被回调。如果栈内为ABCD四个栈顶,A位于栈底,D位于栈顶,再次启动D,如果D启动模式为singleTop,栈内仍为ABCD,如果D是standard,栈内则为ABCDD
4.3
只要Activity在一个栈中存在,那么在该栈中就不会重新创建实例,和singleTop一样也会回调其onNewIntent。当具有singleTask模式的Activity A请求启动后,系统首先会寻找是否存在A想要的任务栈,如果不存在,就重新创建一个任务栈,然后创建A的实例后把A放到栈中。如果存在A所需的任务栈,这时看A是否在栈中有实例存在,如果有,就会把A调到栈顶并调用onNewIntent方法,如果实例不存在,就创建A的实例并把A压入栈中。
4.3
加强的singTask模式,除了具备singleTask模式的所有特性之外,具有此种模式的Activity只能单独地位于一个任务栈中
任务栈:TaskAffinity
- 该参数标识了一个Activity所需要的任务栈的名字,默认情况下,所有Activity所需的任务栈的名字为应用的包名。
- 我们也可以为每个Activity都单独指定TaskAffinity属性,但属性值必须不能和包名相同,否则相当于没有指定。
- TaskAffinity属性主要和singleTask启动模式或者allowTaskReparenting属性配对使用,在其他情况下没有意义
- 任务栈分为前台任务栈和后台任务栈,后台任务栈中的Activity位于暂停状态,用户可以通过切换后台任务栈再次调到前台
- TaskAffinity和allowTaskReparenting结合时。有A和B两个应用,A启动了B的一个Activity C,然后按Home键回到桌面,再点击B的桌面图标,这个时候并不是启动了B的主Activity,二是重新显示了被应用A启动的Activity C。即C从A的任务栈转移到了B的任务栈中。
TaskAffinity使用: 通过AndroidMainfest为Activity指定任务栈 android:taskAffinity="taskName"
如何给Activity指定启动模式,两种方法:
第一种通过AndroidMainfest为Activity指定启动模式,android:launchMode="singleTask"
第二种通过java代码 Intent 中设置标志位为Activity指定启动模式 Intent intent = new Intent(A.this,B.class) ;
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
第二种优先级高于第一种,若两种同时存在,以第二种为准
五. IntentFilter的匹配规则
相关文章:https://blog.csdn.net/xiaohanluo/article/details/52637520
启动Activity分为两种,显示调用和隐式调用。显示调用即需要明确指定被启动对象的组件信息,包括包名和类名,而隐式调用则不需要明确指定组件信息。如果一个Intent既是显示又是隐式则以显式调用为主。
隐式调用需要Intent能够匹配目标组件的IntentFilter中设置的过滤信息action,category,data。
一个过滤列表intent-filter中的action,category,data可以有多个,所有的action,category,data分别构成不同类别。只有一个Intent同时匹配action类别,category类别,data类别才算完全匹配,只有完全匹配才能成功启动目标Activity
一个Activity可以有多个intent-filter,一个Intent只要能匹配任何一组intent-filter即可成功启动对应的Activity。
action的匹配规则:要求Intent中的action存在且必须和过滤规则中的其中一个action相同。actoin区分大小写。
category匹配规则:系统在调用startActivity或者startActivityForResult的时候会默认为Intent加上"android.intent.category.DEFAULT"这个category。
通过隐式调用方式启动一个Activity的时候,需要做一下判断,看是否有Activity能够匹配我们的隐式Intent,如果不做判断,当没有匹配的时候就会崩溃报错。判断的方式有两种:采用PackageManager的resolveActivity方法或者Intent的resolveActivity方法,如果找不到匹配的Activity就会返回null。
不含有DEFAULT这个category的Activity是无法接收隐式Intent的。
<action android:name="android.intent.action.MAIN"/>
<category android:name="android.intent.category.LAUNCHER"/>
这两者共同作用是用来标明这是一个入口Activity并且会出现在系统的应用列表中,少了任何一个都没有实际意义,也无法出现在系统的应用列表中,即二者缺一不可。
我们有时在一个应用中,不小心给两个以上的Activity添加了这两个东西,我们就会发现桌面上就会有两个以上该应用的图标。
网友评论