Activity

20180314更新。
再次梳理了一遍,更详细清晰,见https://github.com/xwpeng/AndroidArt的chapter1

@Deprecated
本文是在读任玉刚的《Android开发艺术探索》上自己的总结,基础概念会略去,主要总结我觉得重要的地方,和书中内容测试后自己的见解。
也想着我以后关于Activity的一些问题都记录在这里。
Activity官方文档
github我测试的demo

正常生命周期注意点

  1. 要有前台/后台可见的概念。
    能与用户直接交互了是前台,能看见不能交互是后台。
    onPause使前台可见结束,onStop使后台可见结束。
    onStart使后台可见,onResume使可交互。
    前台可理解为可见且可交互。
  2. 对于activity优先级可以这样理解:
    可交互>可见不可交互>不可见未destroy
    对于资源回收或者说强杀,感觉粒度是进程而不是组件。
  3. onPause:可停止动画,保存一些数据,非耗时。
  4. onPause执行完新的Activity才能onResume 。所以不能在onPause中做重量(耗时操作),避免b要等待a onStop也尽量不做耗时操作,在onDestroy中做.有些生命周期的回调会受到前一个Activity阻塞。看具体情况去分析。
  5. 从a Activity到b Activity的时候,调onPause,onStop。
    如果b是透明的主题,a不会调onStop。
    透明activity底部弹窗样式:
    <style name="bottom_window_style" parent="Theme.AppCompat.Light">
    <item name="android:windowBackground">@android:color/transparent</item>
    <item name="android:windowIsTranslucent">true</item>
    <item name="android:windowFrame">@null</item>
    <item name="android:windowIsFloating">true</item>
    <item name="android:backgroundDimEnabled">true</item>
    </style>
    熄屏先b的onPause,再a的onStop,再b的onStop。回来也是a优先,有阻塞效应。
  6. onRestart会调onStart。
  7. 注意熄屏,按Home的会触发回调。

Activity的销毁重建的回调

横竖屏Activity的销毁重建,内存不足被杀之类都属于Activity异常结束。我认为被杀可能没有机会走回调方法了,系统的结束清理整个进程的时间极短。

  1. onPause/onStop/onDestory会被调用,onSaveInstance会再onStop之前被调用。被重建的时候在onCreate和onRestoreInstance中取得保存的数据进行恢复。onRestoreInstance在onStart之后调用。官方文档建议onRestoreInstance恢复数据。oncrete中要判断非空
  2. 销毁重建的时候系统默认做恢复工作,比如输入文本内容,ListView滑动位置。是因为这些View中有onSaveInstance/onRestoreInstance方法。EditText继承自TextView。


    TextView中onSaveInstance

    流程:Activity调用onSaveInstance后委托Window保存数据,Window委托DecorView去保存数据,DecorView通知子元素保存数据。

  3. 不希望旋转android:screenOrientation="portrait"始终竖屏。
  4. 旋转时不希望销毁重建android:configChanges="orientation|screenSize",属性说明参考官方文档。
  5. 强制要求旋转(手机已配置禁止自动旋转)在onCreate中加上
    setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_FULL_SENSOR);
  6. 按Home键或者启动新Activity仍然会单独触发onSaveInstanceState的调用。

启动模式与标记位

  • activity实例由任务栈管理,启动模式为了重用栈中已有实例。
  • standard,singleTop,SingleTask,singleInstance四种启动模式。
  • standard 默认的模式,每次启动都是创建新的实例。
  • singleTop 某Activity位于栈顶,再启动它的时候直接复用,onNewIntent被回调。不是在栈顶,依然会创建新的实例。
  • singleTask 任务栈内复用。如果没有相应任务栈/栈中没有此实例,创建栈/把Activity实例化压入。如果有任务栈且任务栈中有此实例,把实例调到栈顶并回调它的onNewIntent.
  • singleInstance 加强的singleTask,单独占据一个任务栈。
    adb shell dumpsys activity 查看手机任务栈信息。
  • 用非activity类型的context启动一个Activity,Activity要带FLAG_ACTIVITY_NEW_TASK标记,或者给intent设置此标志。参考进阶式Context
  • 启动模式设置分两种:manifest指定launchMode,intent.addFlag;第一种无法设置CLEAR_TOP,第二种无法设置singleInstance
  • 标记FLAG_ACTIVITY_NEW_TASK,效果同singleTask
    FLAG_ACTIVITY_SINGLETOP,效果同singleTop
    FLAG_ACTIVITY_CLEAR_TOP。只能通过intent.addFlag设置。sigleTask默认有此标记。
    具有此标记位的Activity,当它启动时,在同一个任务栈中所有位于它的上面的Activity都要出栈。这个标记位一般会和singleTask启动模式一起出现,在这种情况下,被启动Activity的实例如果已经存在,那么系统会调用它的onNewIntent.如果被启动的Activity采用standard模式启动,那么连同它之上的Activity都要出栈,系统会创建新的Activity实例并放入栈顶。
    FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS:同android:excludeFromRecents="true"。使应用不出现在长按home的应用历史使用列表,亲测:需且应用的launchActivity设置生效。

IntentFilter匹配规则总结

官方文档

  • 总体匹配规则:
    一个intent-filter分action,categroy(类别),data三种标记。
    Intent指意图
    Intent中要有categroy + action/data,categroy用系统的。IntentFilter中DEFAULT必须有。

如果三者都有,三个同时匹配成功IntenFilter才匹配成功。
intentfilter可以有多组,匹配任一组就能启动Activity。

  • action匹配规则
    action标记name值自定义的字符串,区分大小写。
    可以有多个action标记,任一匹配即action匹配成功。
    -category规则
    IntentFilter中<categoryandroid:name="android.intent.category.DEFAULT" />这个标记必须有。因为系统在startActivity的时候会默认加上。
    我们在定义我们的规则的时候就用android.intent.category.DEFAULT" 与BROWSER。
    其他类型还存在疑惑,自定义的不行。
  • data匹配规则
    data的schem:https:/,http:/,file:/(两根斜杠)就是url的schem。可以自定义如:xwpeng://
    data的host:为schem后开始到第一根斜杠中间部分,如url xwpeng://blog/xxx/xxx,blog为host
    在intent-filter中只用写xwpeng,blog。斜杠冒号不用写出
    mimeType:指定数据类型,如“image/png”,这中默认的schem只能是file://与content://,你加其他的不行。
    Intent的用setData与setType单独匹配。如果都要设置使用setDataAndType,因为setData会将type置null。setType也会将Uri置null。
  • 最后
    intentfilter与Intent的过滤匹配只能是一般规律总结,用到的时候再灵活应用,多try。用到地方不多,能用显示就用显示。如果涉及到SDK或者Module才会用隐式。官方是建议Service启动绑定尽量用显示。
    BroadcastReceiver也适用匹配过滤。
    隐式启动Activity如果找不到匹配的就会崩溃,如果不确定对应Activity是否存在(情况极少把),用PackageManger与Intent的resolveActivity去判断是否返回null,不是null返回最佳匹配。PackageManger的queryIntentActivity返回所有匹配者的信息。
code.png

相关拓展:Android应用程序的Activity启动过程简要介绍和学习计划

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • Activity 一、四种形态 运行状态: 当 Activity 处于栈的顶层,可见,并可与用户进行交互 onRe...
    任教主来也阅读 5,762评论 1 10
  • 转载注明出处:http://www.jianshu.com/p/c2c2ee4eb48a 1. 简介 本篇不针对于...
    王三的猫阿德阅读 7,114评论 2 5
  • 本篇博客是笔者看过《Android开发艺术探索》才写的,有些是借鉴了此本书的内容,当有些内容进行了精进。 Acti...
    ChenHaHa哈哈阅读 8,795评论 1 27
  • 一本android开发程序员必读的一本书,感谢任玉刚大佬的分享 一、Activity的生命周期全面分析 典型情况下...
    kongjn阅读 3,638评论 1 3
  • 一个人感受到自己的坚强时,一定是受了很大的委屈,在乎的人都在那里喋喋不休,不巧的是,他们多说一句都会觉得心烦,有很...
    陈雨啊阅读 2,578评论 0 0

友情链接更多精彩内容