Android知识点回顾之Activity基础

Activity生命周期

Activity的生命周期包括onCreate(),onRestart(),onStart(),onResume(),onPause(),onStop(),onDestroy()。其相互转化的过程如下图所示。左边的图是单个Activity的生命周期回调情况。右边的是当前Activity跳转到另外一个Activity时,两者相关生命周期回调先后顺序的情形。


图1:Activity生命周期

onCreate()

表示Activity正在被创建

此方法在整个生命周期中只会被调用一次。可以在这里做一些初始化的操作。

参数savedInstanceState保存Activity因为异常情况而被销毁前的状态,可以利用此参数做一些数据恢复的操作。

onStart()

表示Activity正在被启动。此时Activity已经可见,即将进入前台界面,但是还不能喝用户进行交互。

onResume()

表示Activity已经启动完成,进入到了前台,可以同用户进行交互了。需要注意的是此方法在Activity的整个生命周期中可能会被多次调用到。

onPause()

表示Activity正在被停止。

可以在这里释放系统资源,动画的停止

不宜在此做耗时操作,因为此方法结束后会调用新Activity的onCreate(),onStart(),onResume(),耗时操作会影响到新Activity的显示

onStop()

当Activity不可见的时候回调此方法。

需要在这里释放全部用户使用不到的资源。

可以做较重量级的工作,如对注册广播的解注册,对一些状态数据的存储

此时Activity还不会被销毁掉,而是保持在内存中,但随时都会被回收。

onDestroy()

Activity即将被销毁。

此时会释放掉所有占用的资源。

通常的:

onCreate()和onDestroy()成对存在

onStart()和onStop()成对存在

onResume()和onPause()成对存在

Activity状态的保存和恢复

状态的保存

当Activity 非正常退到后台时,就会调用onSaveInstanceState()方法。非正常的情况包括:有新的Activity启动,灭屏,未设置保存状态的横竖屏切换,按home键到桌面,切换到最近任务列表。正常情况是用户点击返回键,程序调用finish()方法,此时不会回调onSaveInstanceState()方法。

调用的次序为:onPause()->onSaveInstanceState()->onStop()

可对一些状态进行保存,

static final String STATE_LISTVIEW_CURRENT_POSITION = "list_view_current_position";
...


@Override
public void onSaveInstanceState(Bundle savedInstanceState) {
    // 保存当前ListView的位置
    savedInstanceState.putInt(STATE_LISTVIEW_CURRENT_POSITION , mCurrentPosition);

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

系统会为View自动保存相关的状态,前提是View要在布局文件里设置唯一的id值,即android:id属性。

由于onSaveInstanceState()方法不能保证百分百调用到,所以只能做一些临时状态信息的存储,如果要做持久化的操作,需要在onStop()里进行。

状态的恢复

当Activity在被销毁后重新创建时,可以从onCreate(Bundle savedInstanceState)方法中的savedInstanceState参数恢复之前保存的状态,也可以从onRestoreInstanceState(Bundle savedInstanceState)方法中恢复。两种方法的效果都是一样的,只是前者需要判空操作,后者不需要

onCreate(Bundle savedInstanceState)的恢复:

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState); 

   //如果不为空的话,则说明有可恢复的状态
    if (savedInstanceState != null) {        
        mCurrentPosition= savedInstanceState.getInt(STATE_LISTVIEW_CURRENT_POSITION );
    } else {
        //初始化
    }
    ...
}

onRestoreInstanceState(Bundle savedInstanceState)的恢复:

//调用此方法,savedInstanceState参数必不为空
public void onRestoreInstanceState(Bundle savedInstanceState) {
    super.onRestoreInstanceState(savedInstanceState);
    mCurrentPosition= savedInstanceState.getInt(STATE_LISTVIEW_CURRENT_POSITION );
}

Activity启动模式

四种启动模式

Activity的启动模式可以允许你的Activity拥有多少个实例。

有两种方法可以设置:1、在manifest文件中设置。2、用Intent的flag标识设置

manifest方式

在minifest中对应的Activity设置如下四个中的一个值。

...
<activity 
  ...
   android:launchMode=["standard" | "singleTop" | "singleTask" | "singleInstance"]
  ...
/>
...

standard

默认值,多实例模式。

每启动一次,都会创建一个新的Activity实例。

并且新实例可以在不同的任务栈上创建,相同的任务栈可以创建多个实例。

启动的生命周期为:onCreate()->onStart()->onResume()

需要注意的是,此模式下启动的实例所在的任务栈,为启动它的那个Activity所属的任务栈。比如A属于task1,A启动了B,B的模式为standard,则B所在的任务栈为task1

singleTop

栈顶复用模式

如果任务栈顶已经存在需要启动的目标Activity,则直接启动,并会回调onNewIntent()方法,生命周期顺序为:
onPause() ->onNewIntent()->onResume()

如果任务栈上顶没有需要启动的目标Activity,则创建新的实例,此时生命周期顺序为:
onCreate()->onStart()->onResume()

两种情况如下图,从图中可以看出,此模式下还是会出现多实例,只要启动的目标Activity不在栈顶的话。

singleTop启动模式,A为standard模式,B为singleTop模式

singleTask

栈内复用模式,一个任务栈只能有一个实例。

有几种情况:

  • 当启动的Activity目标任务栈不存在时,则以此启动Activity为根Activity创建目标任务栈,并切换到前面


    D为singleTask模式
  • 当启动的Activity目标任务栈存在,启动的Activity不存在时,则直接在目标任务栈上创建Activity


    D的启动模式为singleTask,并且taskAffinity为task2
  • 当启动的Activity存在时,则会直接切换到Activity所在的任务栈,并且任务栈中在Activity上面的所有其他Activity都出栈(调用destroy()),此时启动的Activity位于任务栈顶,并且会回调onNewIntent()方法。


    左边的DtaskAffinity为task2的情况,右边D的taskAffinity为task1的情况

singleInstance

和singleTask模式类似,只不过独占其所在的任务栈,比如A为singleInstance模式,其所需的任务栈为task1,则task1只有A一个Activity,由A启动的Activity会另起一个task。而后续启动此Activity,都不会再创建新的Activity。

Intent的方式

除了可以在manifest中设置Activity的启动模式,也可以通过设置Intent的flag标识来设定Activity的启动模式。

常用的有:FLAG_ACTIVITY_NEW_TASK,FLAG_ACTIVITY_SINGLE_TOP,FLAG_ACTIVITY_CLEAR_TOP

FLAG_ACTIVITY_NEW_TASK

启动Activity时,如果不存在Activity的实例,则会以此Activity为根Activity创建新的任务栈,如果存在的话则直接切换到对应的Activity实例,并回调onNewIntent()方法。相当于“singleTask”启动模式。

FLAG_ACTIVITY_SINGLE_TOP

相当于“singleTop”模式

FLAG_ACTIVITY_CLEAR_TOP

设置此标识的Activity在启动时,如果当前的任务栈内存在此Activity实例,则跳转到此实例,并清除掉在此实例上面的所有Activity实例,此时此Activity实例位于任务栈的栈顶

Activity间的数据传递

Activity直接的数据传递,一般使用Intent就够了。

可以使用Intent.putExtra()方法

把值传给待启动的Activity

//MianActivity
 Intent intent = new Intent(this, SecondActivity.class);
 intent.putExtra("data_key", "abcd");
 
 Bundle data = new Bundle();
 data.putString("key1","value1");
 intent.putExtra("data",data);
...
 startActivity(intent);
 
...
//SecondActivity
...
protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.my_activity_layout);
        Intent intent = getIntent();
        Toast.makeText(this,intent.getStringExtra("data_key"),Toast.LENGTH_LONG).show();

        Bundle data = intent.getBundleExtra("data");
        String key1 = data.getString("key1");
        Toast.makeText(this,key1,Toast.LENGTH_LONG).show();
 }
...

把值返回给启动它的Activity

Intent intent = new Intent(MainActivity.this,SecondActivity.class);

 startActivityForResult(intent, REQUEST_CODE);
  
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
 super.onActivityResult(requestCode, resultCode, data);
 String str = data.getStringExtra("data");
 if(REQUEST_CODE == requestCode){
    if(SecondActivity.RESULT_CODE == resultCode){
      ...
    }
 }
 
 ...
}
...
//SecondActivity
...
 Intent intent = new Intent();
 intent.putExtra("data", "result_value");
 setResult(RESULT_CODE, intent);
 finish();
...
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,240评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,328评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,182评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,121评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,135评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,093评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,013评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,854评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,295评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,513评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,678评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,398评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,989评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,636评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,801评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,657评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,558评论 2 352

推荐阅读更多精彩内容