Android 四大组件之Activity

Android 四大组件之Activity

参考文章

小专栏·重学安卓

Google开发者平台

Activity概述

在 Android 平台上,Activity 是应用的基本构建块之一。充当用户与应用互动的入口点,同时也决定了用户在应用内部或应用之间导航的方式。

...

通常,一个 Activity 实现应用中的一个屏幕。大多数应用包含多个屏幕,这意味着它们包含多个 Activity,应用中的各个 Activity 协同工作形成统一的用户体验。

                                                                                                                          ——Google官方文档

Activity生命周期

启动activity的执行流程:onCreate()-->onStart()-->onResume()

销毁activity的执行流程:onPause()-->onStop()-->onDestroy()

aicitity启动流程图
  • onCreate()

    必须实现此回调,它会在系统创建您的 Activity 时触发。

    要做的操作:①恢复数据savedInstanceState;(如果activity在销毁前有保存的话)

                     ②确定布局`setContentView()`;**(必须实现)**
    
                     ③初始化需要用到的控件。
    
  • onStart()

    onCreate() 退出后,Activity 将进入“已启动”状态,并对用户可见。此回调包含 Activity 进入前台与用户进行互动之前的最后准备工作。

    这里重学专栏这篇文章讲的很透彻,对此处“启动”和“可见”有疑惑的同学可以看看。大致意思是将Activity类比为PC桌面操作系统中的窗口,同时一个窗口对应一个进程。

    • 焦点窗口/前台:鼠标点击过的窗口,置于最顶层;对应前景进程

    • 可见窗口:被焦点窗口压着的那个窗口,鼠标不移过去不能操作,但可见;对应可见进程

    这里的“可见”和接着下文中的“获取焦点”就可以理解成上述的两个状态了。

  • onResume()

    这个方法表示界面已经来到了前台,并获取焦点。这是应用与用户互动的状态。应用会一直保持这种状态,直到某些事件发生,让焦点远离应用。(比如接到来电、用户点击到另一个Activity 或 设备屏幕关闭)

    当发生中断事件时,Activity 进入“已暂停”状态,系统调用 onPause() 回调。

  • onPause()

    系统将此方法视为用户将要离开您的 Activity 的第一个标志,此方法表示 Activity 不再位于前台,进入上文中的“可见窗口”状态。之后,系统会保留此状态,直到界面恢复onResume 或 变为完全不可见onStop()

    • 要做的操作: 释放系统资源、传感器手柄等,不再被用户所需要的任何消耗型资源。

    • 不要做的操作:不应该做任何耗时操作,如保存应用或用户数据、进行网络调用或执行数据库事务等。

  • onStop()

    如果您的 Activity 不再对用户可见,说明其已进入“已停止”状态,因此系统将调用 onStop() 回调。系统调用的下一个回调将是 onRestart()(如果 Activity 重新与用户互动)或者 onDestroy()(如果 Activity 彻底终止)。

    应做操作:释放或调整在应用对用户不可见时的无用资源(如暂停动画效果,或从精确位置更新切换到粗略位置更新)

  • onDestroy()

    系统会在销毁 Activity 之前调用此回调。

    onDestroy() 回调应释放先前的回调(例如 onStop())尚未释放的所有资源。

页面跳转生命周期示意图

横竖屏切时的生命周期

此时的生命周期跟清单文件里的配置有关系。

  • 不设置 Activity 的 android:configChanges时,切屏会重新调用各个生
    命周期默认首先销毁当前 activity,然后重新加载。
  • 设置 Activity 的 android:configChanges="orientation|screenSize"时,切屏不会重新调用各个生命周期,只会执行 onConfigurationChanged方法。

启动模式

  • standard(默认)

    标准模式。跳转一个Activity便在栈中新建一个。

  • singleTop

    栈顶复用模式。若跳转的Activity正好是当前Activity(栈顶),则不会在栈中新建Activity。

    流程为:onPause -->onNewIntent-->onResume。

  • singleTask

    栈内复用模式。若跳转的Activity在任务栈中存在(打开过),则不会创建新窗口,而是将此Activity之上的栈全部销毁,直到该Activity变为栈顶。

  • singleInstance

    该 Activity 始终是其任务唯一的成员;由该 Activity 启动的任何 Activity 都会在其他的任务中打开。(可以将该Activity视为一个独立的app【任务】)

image-20210223095127997.png

状态保存

一般情况下,开发者无需关注Activity状态数据的保存,Android系统行为与我们预期相符。

如页面跳转(A跳转到B)中,A调用onPause和onStop方法后,实例仍存在于内存之中,由B返回到A后,状态和数据信息依然保存。B调用onDestroy方法,B中的状态和数据被销毁。在这些情况下,用户的期望与系统行为相匹配,无需完成任何额外工作。

所以此处状态保存指的是某些特殊情况:

如果系统因系统限制(例如配置变更或内存压力)而销毁 Activity,虽然实际的 Activity 实例会消失,但系统会记住它曾经存在过。如果用户尝试回退到该 Activity,系统将使用一组描述 Activity 销毁时状态的已保存数据新建该 Activity 的实例。

这种情况下,Activity在被销毁时会调用onSaveInstanceState()方法,可以在这个方法里面进行数据的保存:

@Override
public void onSaveInstanceState(Bundle savedInstanceState) {
    // Save the user's current game state
    savedInstanceState.putInt(STATE_SCORE, currentScore);
    savedInstanceState.putInt(STATE_LEVEL, currentLevel);

    // Always call the superclass so it can save the view hierarchy state
    super.onSaveInstanceState(savedInstanceState);
}

由于是用的bundle保存数据,所以只能够保存轻量级的数据。

Bundle 对象并不适合保留大量数据,因为它需要在主线程上进行序列化处理并占用系统进程内存。如需保存大量数据,您应组合使用持久性本地存储、onSaveInstanceState() 方法和 ViewModel 类来保存数据,正如保存界面状态中所述。
... ...
如需保存持久性数据(例如用户首选项或数据库中的数据),您应在 Activity 位于前台时抓住合适机会。如果没有这样的时机,您应在执行 onStop() 方法期间保存此类数据。

在Activity被重新启动时,可以在onCreate()或者onRestoreInstanceState()方法中获取到包含实例状态信息的相同 Bundle

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState); // Always call the superclass first

    // Check whether we're recreating a previously destroyed instance
    if (savedInstanceState != null) {
        // Restore value of members from saved state
        currentScore = savedInstanceState.getInt(STATE_SCORE);
        currentLevel = savedInstanceState.getInt(STATE_LEVEL);
    } else {
        // Probably initialize members with default values for a new instance
    }
    // ...
}

onRestoreInstanceState()调用在onStart()之后,仅当有bundle数据返回时才会调用,所以onRestoreInstanceState中不用判空操作。

public void onRestoreInstanceState(Bundle savedInstanceState) {
    // Always call the superclass so it can restore the view hierarchy
    super.onRestoreInstanceState(savedInstanceState);

    // Restore state members from saved instance
    currentScore = savedInstanceState.getInt(STATE_SCORE);
    currentLevel = savedInstanceState.getInt(STATE_LEVEL);
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,992评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,212评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,535评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,197评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,310评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,383评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,409评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,191评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,621评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,910评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,084评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,763评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,403评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,083评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,318评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,946评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,967评论 2 351