【Android】9.0活动的生命周期(二)——实际代码演示

1.0 新建一个项目ActivityLifeCycleTest。

新建两个活动NormalActivity和DialogActivity,勾选“Generate Layout File”。

文件目录如下:

image
2.0 修改主活动布局,增加两个按钮控件,以响应新建好的这两个活动,activity_main.xml:
<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">

    <Button
        android:id="@+id/start_narmal_activity"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="开始NormalActivity活动"
        tools:layout_editor_absoluteX="0dp"
        tools:layout_editor_absoluteY="40dp" />

    <Button
        android:id="@+id/start_dialog_activity"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginTop="84dp"
        android:text="开始DialogActivity活动"
        app:layout_constraintTop_toTopOf="@id/start_narmal_activity"
        tools:layout_editor_absoluteX="0dp" />
</android.support.constraint.ConstraintLayout>
3.0 在两个活动布局文件中添加文字打印说明,activity_normal.xml:
<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".NormalActivity">

    <TextView
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="这是一个普通活动"/>
</android.support.constraint.ConstraintLayout>

activity_dialog.xml:

<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".DialogActivity">
<TextView
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:text="这是一个对话框式活动"/>
</android.support.constraint.ConstraintLayout>
4.0 这里需要修改一下AndroidMainifest.xml中关于活动DialogActivity的内容,以把它变成对话框式的活动主题:
<!--android:theme是给当前活动指定主题的,-->
        <!--android内置了很多主题可以选择,-->
        <!--当然我们也可以定制自己的主题-->
        <!--当前Dialog主题主要就是使用对话框式的主题-->
        <activity
            android:name=".DialogActivity"
            android:theme="@style/Theme.AppCompat.Dialog"></activity>
        <activity android:name=".NormalActivity" />
5.0 最后的大头,主活动MainActivity.java,通过重写6个回调方法(onCreate()方法在活动创建时自动重写了,而且也必须重写),通过logcat调试监控窗口来体现方法的调用:
package com.example.activitylifecycletest;

import android.content.Intent;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.Toast;

public class MainActivity extends AppCompatActivity {
    public static final String TAG = "主活动";

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        Log.d(TAG,"onCreate");
        setContentView(R.layout.activity_main);
        Button startNormalActivity = (Button) findViewById(R.id.start_narmal_activity);
        Button startDialogActivity = (Button) findViewById(R.id.start_dialog_activity);
        startNormalActivity.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent intent = new Intent(MainActivity.this,NormalActivity.class);
                startActivity(intent);
            }
        });
        startDialogActivity.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent intent = new Intent(MainActivity.this,DialogActivity.class);
                startActivity(intent);
            }
        });
    }

    @Override
    protected void onStart() {
        super.onStart();
        Log.d(TAG,"onStart");
    }

    @Override
    protected void onResume() {
        super.onResume();
        Log.d(TAG,"onResume");
    }

    @Override
    protected void onPause() {
        super.onPause();
        Log.d(TAG,"onPause");
    }

    @Override
    protected void onStop() {
        super.onStop();
        Log.d(TAG,"onStop");
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        Log.d(TAG,"onDestroy");
    }

    @Override
    protected void onRestart() {
        super.onRestart();
        Log.d(TAG,"onRestart");
    }
}
6.0 好了,运行,体验一下:

6.1运行程序run,进入活动MainActivity页面,这时logcat已经在打印东西了,调到Verbose级别的bug监测(以下监测内容都是这个级别):

image

可以看到:

  • 一开始时 onCreate() 打印,活动一创建自动有,每个活动都会重写这个方法。
  • 活动MainActivity目前处于可见状态,入栈:
  • 此时调用onStart() 该方法在活动由不可见变为可见时调用。
  • 接着调用onResume() 该方法在活动准备好和用户进行交互的时候调用,而且活MainActivity的确位于返回栈的栈顶,并处于运行状态。

此时进入如下界面:

image

6.2 然后,点击第一个按钮“开始NORMALACTIVITY活动”:

image

此时logcat刷新为:

image

可以看到:

  • 此时活动MainActivity将处于不可见状态,而活动NormalActivity将处于可见状态,并加入栈顶;于是onPause() 该方法在系统准备去启动或者恢复另一个活动的时候调用。活动NormalActivity此时进入可见状态。
  • 接着调用onStop()方法,在活动MainActivity完全不可见的时候调用,它和onPause() 方法的主要区别是,如果启动的新活动是一个对话框式的活动,那么onPause() 方法会得到执行,而onStop()方法并不会执行。

6.3 这时已经进入NormalActivity活动界面,点击Back键:

image

此时logcat刷新为:

image

可以看到:

  • 此时活动NormalActivity将被销毁,出栈,而活动MainActivity将变为栈顶,由不可见变为可见状态,可见先调用onRestart()方法先激活活动MainActivity,再调用onStart()方法,在活动由不可见变为可见时调用,进入可见生存期,最后调用onResume()方法,活动MainActivity做好准备,和用户进行交互,进入前台生存期,也在可见生存期内。
  • 但是我们可以看到,并没有调用 onDestroy()方法去销毁活动,虽然此时活动NormalActivity已经出栈。
7.0 这时已经进入MainActivity主活动界面,点击第二个按钮“开始DIALOGACTIVITY活动”:
image

此时logcat刷新为:

image

可以看到:

  • 仅调用 onPause()方法,使系统去启动活动MialogActivity。因为启动的是一个对话框式活动,活动MainActivity并不会被执行onStop()方法。
  • 除此之外,并没有执行其他回调函数,说明活动MainActivity退出前台生存期,但是处于可见生存期内。
8.0 这时进入如下画面:
image

如上图箭头,点击Back键:

此时logcat刷新为:

image

可以看到:

  • 对话框式活动直接返回上一活动,并不需要执行onStop()方法,因为活动MainActivity并不处于完全不可见状态。直接返回调用onResume()方法。
  • onResume()方法在活动MainActivity准备好和用户进行交互的时候调用。
9.0 这时进入如下界面:
image
10.0 点击Back键:

此时logcat刷新为:

image

可以看到:

  • 程序立马执行onPause()方法,释放CPU资源,保存关键数据,前台生存期结束。
  • 活动MainActivity进入完全不可见状态,所以调用onStop()方法,可见生存期结束。
  • 最后整个活动被销毁,调用onDestroy,一个完成生存期结束。
  • 可见onDestroy()方法调用,会在整个app应用程序退出时调用。
11.0 最后退出软件:
image
12.0 再结合官方的活动生命示意图,一下就理解了:
image
13.0 还有一个问题:当用户在活动A的基础上启动了活动B,这个时候由于内存不足,活动A被回收,这时候当从活动B返回时,活动A将重新创建,并丢失待机时可能存在的数据,这样严重影响用户体验的。

onSaveInstanceState()回调方法可以解决活动被回收时临时数据得不到保存的问题。

这个方法能够保证在活动被收回之前一定会被调用。

在MainActivity.java中添加如下代码就可以实现:

//    该方法携带一个Bundle类型的参数,该参数提供一系列方法用于保存数据
//    可以使用putString()方法保存字符串
//    使用putInt()方法保存整型数据
    @Override
    public void onSaveInstanceState(Bundle outState, PersistableBundle outPersistentState) {
        super.onSaveInstanceState(outState);
        String tempData = "你刚刚编辑的临时东西";
//        使用putString()方法保存字符串
//        需要提供两个参数,一个是键名,第二个才是需要保存的内容
        outState.putString("data_key",tempData);
    }

数据保存下来,恢复怎么办?

其实我们一直使用的onCreate()方法,自带有一个Bundle类型的参数,这个参数一般情况是null,所以只要我们通过onSaveInstanceState()方法保存数据,这个参数就会带着之前保存的全部数据。

我们只需要取出来即可,修改MainActivity方法的onCreate()方法:

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

推荐阅读更多精彩内容