安卓活动的生命周期(重点演示、叙述:完整的项目生命历程演示)


ps本文为学习《第一行代码:Android(第2版)》.pdf时所记,内容有所参照。

活动的生命周期

(i)返回栈

Android是使用任务(Task)来管理活动的,一个任务就是一组存放在栈里的活动的集合,这个栈,也被称为返回栈(Back Stack)。想象一下栈后进先出的特性,再加上下面这幅图,就不难理解了。
在这里插入图片描述

(ii)活动状态

每个活动在其生命周期中最多可能会有4种状态。

1运行状态

当一个活动位于返回栈的栈顶时,这时活动就处于运行状态。系统最不愿意回收的就是处于运行状态的活动,因为这会带来非常差的用户体验。

2暂停状态

当一个活动不再处于栈顶位置,但仍然可见时,这时活动就进入了暂停状态。你可能会觉得既然活动已经不在栈顶了,还怎么会可见呢?这是因为并不是每一个活动都会占满整个屏幕的,比如对话框形式的活动只会占用屏幕中间的部分区域,你很快就会在后面看到这种活动。处于暂停状态的活动仍然是完全存活着的,系统也不愿意去回收这种活动(因为它还是可见的,回收可见的东西都会在用户体验方面有不好的影响),只有在内存极低的情况下,系统才会去考虑回收这种活动。

3停止状态

这种活动保存相应的状态和成员变量,但是这并不是完全可靠的,当其他地方需要内存时,处于停止状态的活动有可能会被系统回收。

4销毁状态

当一个活动从返回栈中移除后就变成了销毁状态。系统会最倾向于回收处于这种状态的活动,从而保证手机的内存充足。

(iii)活动的生存期

  Activity类中定义了7个回调方法,覆盖了活动生命周期的每-一个环节, 下面就来- -- 介绍这7个方法。
  口onCreate()。这个方法你已经看到过很多次了,每个活动中我们都重写了这个方法,它
  会在活动第--次被创建的时候调用。你应该在这个方法中完成活动的初始化操作,比如说加载布局、绑定事件等。
  口onStart()。这个方法在活动由不可见变为可见的时候调用。
  口onResume()。这个方法在活动准备好和用户进行交互的时候调用。此时的活动一定位于
  返回栈的栈顶,并且处于运行状态。
  口onPause()。这个方法在系统准备去启动或者恢复另一个活动的时候调用。我们通常会在这个方法中将一些消耗CPU的资源释放掉,以及保存一些关键数据,但这个方法的执行速度一定要快,不然会影响到新的栈顶活动的使用。
  口onStop()。这个方法在活动完全不可见的时候调用。它和onPause( )方法的主要区别在
  于,如果启动的新活动是一个对话框式的活动,那么onPause( )方法会得到执行,而,onStop()方法并不会执行。
  onDestroy()。 这个方法在活动被销毁之前调用,之后活动的状态将变为销毁状态。
  onRestart()。 这个方法在活动由停止状态变为运行状态之前调用,也就是活动被重新启动了。
  
  以上7个方法中除了onRestart()方法, 其他都是两两相对的,从而又可以将活动分为3种生存期。
  
  完整生存期。活动在onCreate()方法和onDestroy()方法之间所经历的,就是完整生存期。一般情况下,一个活动会在onCreate()方法中完成各种初始化操作,而在onDestroy()方法中完成释放内存的操作。

	可见生存期。活动在onStart()方法和onStop()方法之间所经历的,就是可见生存期。在可见生存期内,活动对于用户总是可见的,即便有可能无法和用户进行交互。
	我们可以通过这两个方法,合理地管理那些对用户可见的资源。比如在onStart()方法中对资源进行加载,而在onStop( )方法中对资源进行释放,从而保证处于停止状态的活动不会占用过多内存。

	前台生存期。活动在onResume( )方法和onPause()方法之间所经历的就是前台生存期。在前台生存期内,活动总是处于运行状态的,此时的活动是可以和用户进行交互的,我们平时看到和接触最多的也就是这个状态下的活动。

为了帮助你能够更好地理解,Android 官方提供了一张活动生命周期的示意图
在这里插入图片描述

(iv)体验一下活动的生命周期(重点)

创建了一个项目来详细的体验一下活动的生命周期:
在这里插入图片描述

创建一个empty project(我命名为project_testlife),系统会自动帮我们创建布局,如图红色画线(activity_main.xml)和活动(如图红色画线MainActivity),为了测试我们还需创建如图绿色对勾标记的布局和活动。

代码如下:

1)MainActivity

package com.example.project_testlife;

import androidx.appcompat.app.AppCompatActivity;
import android.content.Intent;
import android.net.Uri;
import android.os.Bundle;
import android.util.Log;
import android.view.Menu;
import android.view.View;
import android.widget.Button;
import android.widget.Toast;
import android.os.Bundle;

public class MainActivity extends AppCompatActivity {

    private static final String TAG = "MainActivity";
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Button startNormalActivity = (Button) findViewById(R.id.but1);
        Button startDialogActivity = (Button) findViewById(R.id.but2);

        startNormalActivity.setOnClickListener(new View.OnClickListener() {
            public void onClick(View v) {
                Intent intent_temp = new Intent(MainActivity.this,NormalAcitivity.class);
                startActivity(intent_temp);
            }
        });

        startDialogActivity.setOnClickListener(new View.OnClickListener() {
            public void onClick(View v) {
                Intent intent_temp = new Intent(MainActivity.this,DialogAcitivity.class);
                startActivity(intent_temp);
            }
        });
    }

    //几个......

    @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");
    }
}

2)activity_main.xml布局

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <Button
    android:id="@+id/but1"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:text="jump to normalactivity"/>

    <Button
        android:id="@+id/but2"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="jump to dialogactivity"/>
</LinearLayout>
  1. 4)NormalAcitivity和DialogAcitivity活动创建就好,不需要写什么代码

5)activity_normal_acitivity.xml布局文件

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="match_parent"
    android:layout_height="match_parent">


    <TextView
        android:id="@+id/nor_textView2"
        android:layout_width="match_parent"
        android:layout_height="61dp"
        android:text="this is normal_activity" />
</LinearLayout>

6)activity_diaog_acitivity.xml布局文件

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="match_parent"
    android:layout_height="match_parent">


    <TextView
        android:id="@+id/dia_textView2"
        android:layout_width="match_parent"
        android:layout_height="61dp"
        android:text="this is dialog_activity" />
</LinearLayout>

7)NormalAcitivity和DialogAcitivity活动对应的布局没有什么不同,NormalAcitivity是一个普通活动;把DialogAcitivity设置成对话框式的活动,这需要在AndroidManifest中完成。

重点关注下面这段语句:

 <activity android:name=".DialogAcitivity"
            android:theme="@style/Theme.AppCompat.Dialog"></activity>

完整的AndroidManifest代码。

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.project_testlife">

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/AppTheme">
        <activity android:name=".DialogAcitivity"
            android:theme="@style/Theme.AppCompat.Dialog"></activity>
        <activity android:name=".NormalAcitivity" />
        <activity android:name=".MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>

</manifest>

至此,该测试项目就完成了,让我们通过日志来观察下活动的生命历程吧!

1.运行项目,执行onCreate(),onStart(),on Resume()方法
在这里插入图片描述
2.点击第一个按钮,启动NormalAcitivity活动,由于NormalAcitivity把MainActivity完全遮住,所以MainActivity执行了onPause() , onStop()两个方法

在这里插入图片描述

3.按下back键返回MainActivity,因为之前MainActivity进入了停止状态,所以执行了onRestart(), onStart(), onResume()三个方法。
在这里插入图片描述
4.点击第二个按钮,启动DialogAcitivity,因为DialogAcitivity没有完全遮住MainActivity,所以MainActivity只是进入暂停状态,并没有停止,所以只执行onPause()
在这里插入图片描述

5.按下back键返回MainActivity,因为之前MainActivity进入了暂停状态,所以执行了onResume()方法。
在这里插入图片描述
6.最后按back退出程序,依次执行onPause(), onStart(), onDestroy(), 最终销毁程序(未配图)

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值