美文网首页
Activity疑难知识点

Activity疑难知识点

作者: 何小送 | 来源:发表于2020-04-23 16:06 被阅读0次

    笔记来源:任玉刚《Android开发艺术探索》

    一. Activity生命周期切换

    Activity生命周期切换过程

    二. 正常情况下的生命周期分析

    2.1.onStart和onResume,onPause和onStop的区别

    onStart和onStop是从Activity \color{red}{是否可见}这个角度来回调的。
    onResume和onPause是从Activity \color{red}{是否位于前台}这个角度来回调的。
    onStart的时候,Activity可见但还在后台(我们看不到),onResume的时候Activity才显示在前台

    2.2. 当前Activity为A,此时用户打开一个新ActivityB,那么A的onPause和B的onResume哪个先执行?

    必须A的onPause执行完成以后,ActivityB才能启动

    三. 异常情况下的生命周期分析

    如横竖屏切换,默认情况下,Activity会被销毁并重新创建,非默认情况下,可以阻止系统重新创建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 \color{red}{standard:标准模式。}
    每次启动一个Activity都会重新创建一个新的实例,不管这个实例是否已经存在。谁启动新Activity,新Activity就运行在启动它的那个Activity的栈中
    4.2 \color{red}{singleTop:栈顶复用模式。}
    如果新Activity已经位于任务栈的栈顶,那么此Activity不会被重建,onCreate,onStart不会被调用,其onNewIntent方法会被回调。如果栈内为ABCD四个栈顶,A位于栈底,D位于栈顶,再次启动D,如果D启动模式为singleTop,栈内仍为ABCD,如果D是standard,栈内则为ABCDD
    4.3 \color{red}{singleTask:栈内复用模式。}
    只要Activity在一个栈中存在,那么在该栈中就不会重新创建实例,和singleTop一样也会回调其onNewIntent。当具有singleTask模式的Activity A请求启动后,系统首先会寻找是否存在A想要的任务栈,如果不存在,就重新创建一个任务栈,然后创建A的实例后把A放到栈中。如果存在A所需的任务栈,这时看A是否在栈中有实例存在,如果有,就会把A调到栈顶并调用onNewIntent方法,如果实例不存在,就创建A的实例并把A压入栈中。
    4.3 \color{red}{singleInstance:单实例模式。}
    加强的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添加了这两个东西,我们就会发现桌面上就会有两个以上该应用的图标。

    相关文章

      网友评论

          本文标题:Activity疑难知识点

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