当前位置: 首页 > news >正文

Android零基础入门第75节:Activity状态和生命周期方法

   前面两期我们学习了Activity的创建和注册、以及启动和关闭,也学会了重写onCraete方法,这些知识在实际开发中远远不够,还需要学习了解更多。

    生命周期就是一个对象从创建到销毁的过程,每一个对象都有自己的生命周期。同样, Activity也具有相应的生命周期。

 

 

一、Activity状态

 

    一个Android应用程序往往包含多个Activity,当Activity处于Android应用中运行时,每个Activity的状态都不一样。Activity的活动状态由Android以Activity栈的形式管理,当前活动的Activity位于栈顶。随着不同应用的运行,每个Activity都有可能从活动状态转入非活动状态,也可能从非活动状态转入活动状态。

    Activity的生命周期中一共有4种状态,分别如下:

1、运行状态(Active or Running) 

    当Activity在屏幕的最前端时,它处于Activity栈顶,是可见的、有焦点的,可以用来处理用户的常见操作,如点击、双击、长按事件等,这种状态称为运行状态,也叫活动状态。

2、暂停状态(Paused) 

    在某些情况下,Activity对用户来说仍然是可见的,但它不再拥有焦点,即用户对它的操作是没有实际意义的。例如,当最上面的Activity没有完全覆盖屏幕或者是透明的,被覆盖的Activity仍然对用户可见,并且存活(它保留着所有的状态和成员信息并保持与Activity管理器的连接)。但当内存不足时,这个暂停状态的Activity可能会被杀死。

3、停止状态(Stopped) 

    当Activity完全不可见时,它就处于停止状态,但仍然保留着当前状态和成员信息。然而这些对用户来说都是不可见的,如果当系统内存不足时,这个Activity很容易被杀死。

4、销毁状态(Killed) 

    当Activity运行结束,或Activity所在的进程结束时,这种状态称为销毁状态,也叫非活动状态。这时Activity已从Activity栈中移除,需要重新启动才可以显示和使用。当系统内存需要被用在其他地方的时候,一个停止状态的Activity被杀掉。

    另外当Activity处于运行状态时,Android会尽可能地保持它的运行,即使出现内存不足的情况,Android也会先杀死栈底部的Activity,来确保可见的Activity正常运行。 

    关于Activity的这四种状态,是可以相互转化的,如下图所示。

    值得注意的时这四种状态中,运行状态和暂停状态是可见的,停止状态和销毁状态是不可见的。

 

 

二、 Activity生命周期

 

    Activity从一种状态转变到另一种状态时会触发一些事件,执行一些回调方法来通知状态的变化,这就是Activity的生命周期。

    Activity的生命周期及其相关方法的回调,如下图所示。

    从上图可以看到在Activity的生命周期中,有如下几个方法被系统回调。

  • onCreate(Bundle savedStatus):创建Activity时被回调。该方法只会被调用一次。

  • onStart():启动 Activity 时被回调。

  • onRestart():重新启动 Activity 时被回调。

  • onResume():恢复 Activity 时被回调。在onStart()方法后一定会回调 onResume()方法。

  • onPause():暂停 Activity 时被回调。

  • onStop():停止 Activity 时被回调。

  • onDestroy():销毁 Activity 时被回调。该方法只会被调用一次。

     为了更好的掌握Activity生命周期及其方法,接下来通过一个实例来学习,主要实现两个Activity之间跳转时生命周期方法变化的过程。

    仍然使用上期创建的ActivitySample工程,为了便于代码管理,这里新建一个activitylifecycle的Module。该Module一共包含了2个Activity,并在AndroidManifest清单文件中配置两个Activity。

    其中第一个Activity的界面布局非常简单,只包含2个按钮,其中第一个按钮的作用主要是启动第二个Activity,第二个按钮的作用是退出当前Activity。第一个Activity对应的布局文件activity_main.xml的代码如下:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:layout_width="match_parent"
              android:layout_height="match_parent"
              android:orientation="vertical">
    <Button
        android:id="@+id/start_normal_activity_btn"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="启动Activity"/>
    <Button
        android:id="@+id/quit_activity_btn"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="退出Activity"/>
</LinearLayout>

    第二个Activity对应的布局文件activity_second.xml的代码如下:

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

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="这是第二个Activity"/>
</LinearLayout>

    第一个MainActivity类主要用于重写Activity的生命周期方法,并在每个方法中打印出Log以便观察,具体代码如下所示:

package com.jinyu.cqkxzsxy.android.activitylifecycle;

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

public class MainActivity extends AppCompatActivity implements View.OnClickListener {
    private static final String TAG = "MainActivity";
    private Button mStartActivityBtn = null;
    private Button mQutiActivityBtn = null;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Log.d(TAG, "onCreate()");

        mStartActivityBtn = (Button) findViewById(R.id.start_normal_activity_btn);
        mQutiActivityBtn = (Button) findViewById(R.id.quit_activity_btn);
        mStartActivityBtn.setOnClickListener(this);
        mQutiActivityBtn.setOnClickListener(this);
    }

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

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

    @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
    public void onClick(View view) {
        switch (view.getId()){
            case R.id.start_normal_activity_btn:
                // 启动普通的Activity
                Intent intent = new Intent(MainActivity.this, SecondActivity.class);
                startActivity(intent);
                break;
            case R.id.quit_activity_btn:
                // 结束该Activity
                MainActivity.this.finish();
                break;
            default:
                break;
        }
    }
}

    为了观察MainActivity停止状态时的生命周期,需要在当前项目中创建第二个SecondActivity,由于不需要对第二个Activity进行界面操作,因此添加activity_second.xml文件即可不需要其他操作。在第二个Activity中同样实现Activity生命周期中的方法,在每个方法中打印Log信息。

package com.jinyu.cqkxzsxy.android.activitylifecycle;

import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;

public class SecondActivity extends AppCompatActivity {
    private static final String TAG = "SecondActivity";
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_second);
        Log.d(TAG, "onCreate()");
    }

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

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

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

    在AndroidManifest.xml文件中注册已经创建好的Activity,即在清单文件中添加一个<activity>结点,指定Activity径名。

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

    <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=".MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN"/>
                <category android:name="android.intent.category.LAUNCHER"/>
            </intent-filter>
        </activity>

        <activity android:name=".SecondActivity" />
    </application>
</manifest>

    上述操作完成后运行程序,首先会显示第一个Activity界面,如下图所示。

    这时在Android Studio的LogCat窗口会打印MainActivity生命周期中的执行方法,如下图所示:

    从上图可以看到,应用程序启动 MainActivity依次调用了 onCreate()、onStart()、onResume(),这个顺序是第一个Activity从创建到显示在前台到用户可点击的过程。

    接下来单击第一个按钮启动第二个Activity。当第一个界面跳转到第二个界面时,LogCat 窗口会打印MainActivity和SecondActivity生命周期中的执行方法,如下图所示:

    从上图可以看到,当跳转到第二个界面时,MainActivity首先失去焦点执行了onPause()方法,然后SecondActivity依次执行了onCreate()、onStart()、onResume()方法从创建到前台可见, 这时 MainActivity执行了onStop()方法。

    现在再观察一下从第二个Activity按返回键回到第一个Activity生命周期的LogCat 信息,如下图所示:

    从上图可以看到,单击返回键之后,SecondActivity同样先执行了onPause()方法,然后 MainActivity执行了 onRestart()、onStart()、onResume()方法,随后SecondActivity才彻底关闭,执行 了 onStop()、onDestory()。在 MainActivity打开 SecondActivity时,MainActivity并没有执 onDestory()方法而是执行了 onStop()方法。因此,从 SecondActivity返回到 MainActivity时,MainActivity执行了 onRestart() 方法。

    单击退出Activity按钮,MainActivity将会结束自己,并且可以在LogCat 窗口看到如下图所示的输出:

    从该示例可以看到,各生命周期方法的调用完全符合前面所讲。

    将Activity的生命周期方法和Activity的四种状态结合起来,用另一种方式表现出来,可以得到下面的示意图。

    关于Activity的生命周期方法就先学到这里,下期再继续学习其他有关Activity的内容。

 

    今天就先到这里,如果有问题欢迎留言一起探讨,也欢迎加入Android零基础入门技术讨论微信群,共同成长!

    如果该系列分享对你有帮助,就动动手指关注、点赞、留言吧,你的互动就是对我最大的鼓励!

   此文章版权为微信公众号分享达人秀(ShareExpert)——鑫鱻所有,若需转载请联系作者授权,特此声明!

 

往期总结回顾:

Android零基础入门第1节:Android的前世今生

Android零基础入门第2节:Android 系统架构和应用组件那些事

Android零基础入门第3节:带你一起来聊一聊Android开发环境

Android零基础入门第4节:正确安装和配置JDK, 高富帅养成第一招

Android零基础入门第5节:善用ADT Bundle, 轻松邂逅女神

Android零基础入门第6节:配置优化SDK Manager, 正式约会女神

Android零基础入门第7节:搞定Android模拟器,开启甜蜜之旅

Android零基础入门第8节:HelloWorld,我的第一趟旅程出发点

Android零基础入门第9节:Android应用实战,不懂代码也可以开发

Android零基础入门第10节:开发IDE大升级,终于迎来了Android Studio

Android零基础入门第11节:简单几步带你飞,运行Android Studio工程

Android零基础入门第12节:熟悉Android Studio界面,开始装逼卖萌

Android零基础入门第13节:Android Studio个性化配置,打造开发利器

Android零基础入门第14节:使用高速Genymotion,跨入火箭时代

Android零基础入门第15节:掌握Android Studio项目结构,扬帆起航

Android零基础入门第16节:Android用户界面开发概述

Android零基础入门第17节:文本框TextView

Android零基础入门第18节:输入框EditText

Android零基础入门第19节:按钮Button

Android零基础入门第20节:复选框CheckBox和单选按钮RadioButton

Android零基础入门第21节:开关组件ToggleButton和Switch

Android零基础入门第22节:图像视图ImageView

Android零基础入门第23节:图像按钮ImageButton和缩放按钮ZoomButton

Android零基础入门第24节:自定义View简单使用,打造属于你的控件

Android零基础入门第25节:简单且最常用的LinearLayout线性布局

Android零基础入门第26节:两种对齐方式,layout_gravity和gravity大不同

Android零基础入门第27节:正确使用padding和margin

Android零基础入门第28节:轻松掌握RelativeLayout相对布局

Android零基础入门第29节:善用TableLayout表格布局

Android零基础入门第30节:两分钟掌握FrameLayout帧布局

Android零基础入门第31节:少用的AbsoluteLayout绝对布局

Android零基础入门第32节:新推出的GridLayout网格布局

Android零基础入门第33节:Android事件处理概述

Android零基础入门第34节:Android中基于监听的事件处理

Android零基础入门第35节:Android中基于回调的事件处理

Android零基础入门第36节:Android系统事件的处理

Android零基础入门第37节:初识ListView

Android零基础入门第38节:初识Adapter

Android零基础入门第39节:ListActivity和自定义列表项

Android零基础入门第40节:自定义ArrayAdapter

Android零基础入门第41节:使用SimpleAdapter

Android零基础入门第42节:自定义BaseAdapter

Android零基础入门第43节:ListView优化和列表首尾使用

Android零基础入门第44节:ListView数据动态更新

Android零基础入门第45节:网格视图GridView

Android零基础入门第46节:列表选项框Spinner

Android零基础入门第47节:自动完成文本框AutoCompleteTextView

Android零基础入门第48节:可折叠列表ExpandableListView

Android零基础入门第49节:AdapterViewFlipper图片轮播

Android零基础入门第50节:StackView卡片堆叠

Android零基础入门第51节:进度条ProgressBar

Android零基础入门第52节:自定义ProgressBar炫酷进度条

Android零基础入门第53节:拖动条SeekBar和星级评分条RatingBar

Android零基础入门第54节:视图切换组件ViewSwitcher

Android零基础入门第55节:ImageSwitcher和TextSwitcher

Android零基础入门第56节:翻转视图ViewFlipper

Android零基础入门第57节:DatePicker和TimePicker选择器

Android零基础入门第58节:数值选择器NumberPicker

Android零基础入门第59节:常用三大Clock时钟组件

Android零基础入门第60节:日历视图CalendarView和定时器Chronometer

Android零基础入门第61节:滚动视图ScrollView

Android零基础入门第62节:搜索框组件SearchView

Android零基础入门第63节:值得借鉴学习的选项卡TabHost

Android零基础入门第64节:揭开RecyclerView庐山真面目

Android零基础入门第65节:RecyclerView分割线开发技巧

Android零基础入门第66节:RecyclerView点击事件处理

Android零基础入门第67节:RecyclerView数据动态更新

Android零基础入门第68节:RecyclerView添加首尾视图

Android零基础入门第69节:ViewPager快速实现引导页

Android零基础入门第70节:ViewPager打造TabHost效果

Android零基础入门第71节:CardView简单实现卡片式布局

Android零基础入门第72节:SwipeRefreshLayout下拉刷新

Android零基础入门第73节:Activity创建和配置

Android零基础入门第74节:Activity启动和关闭

转载于:https://www.cnblogs.com/cqkxzsxy/p/7660156.html

相关文章:

  • Android之String国际化对照表
  • 2.3 js刷新页面所有 我的程序猿之路:第十三章
  • redisCheckMem脚本
  • 搜索引擎技术之概要预览
  • 从BLOG到电子书——把wordpress备份的WXR转为epub
  • 【数据结构栈和队列】之十队列顺序存储结构
  • ORA-00980: synonym translation is no longer valid, version 9
  • [原创] css中的绝对定位和相对定位
  • 页面触底自动加载数据
  • SSL配置及练习
  • 从上百幅架构图中学得半点大型网站建设经验(上)
  • jquery选择器
  • 深切悼念苹果的老大乔布斯
  • css理论
  • windwos7 em控制台使用操作系统用户错误
  • 【个人向】《HTTP图解》阅后小结
  • - C#编程大幅提高OUTLOOK的邮件搜索能力!
  • FastReport在线报表设计器工作原理
  • Java 内存分配及垃圾回收机制初探
  • Java 网络编程(2):UDP 的使用
  • Java比较器对数组,集合排序
  • MobX
  • Mybatis初体验
  • mysql_config not found
  • MySQL-事务管理(基础)
  • nodejs调试方法
  • pdf文件如何在线转换为jpg图片
  • Redux 中间件分析
  • Vue 重置组件到初始状态
  • Web标准制定过程
  • 半理解系列--Promise的进化史
  • 和 || 运算
  • 后端_MYSQL
  • 配置 PM2 实现代码自动发布
  • 使用parted解决大于2T的磁盘分区
  • 系统认识JavaScript正则表达式
  • 掌握面试——弹出框的实现(一道题中包含布局/js设计模式)
  • 1.Ext JS 建立web开发工程
  • 支付宝花15年解决的这个问题,顶得上做出十个支付宝 ...
  • ​Base64转换成图片,android studio build乱码,找不到okio.ByteString接腾讯人脸识别
  • ​queue --- 一个同步的队列类​
  • # Swust 12th acm 邀请赛# [ A ] A+B problem [题解]
  • #我与Java虚拟机的故事#连载03:面试过的百度,滴滴,快手都问了这些问题
  • #我与Java虚拟机的故事#连载19:等我技术变强了,我会去看你的 ​
  • (多级缓存)多级缓存
  • (附源码)ssm高校社团管理系统 毕业设计 234162
  • (一)Mocha源码阅读: 项目结构及命令行启动
  • (原)记一次CentOS7 磁盘空间大小异常的解决过程
  • (转)Oracle存储过程编写经验和优化措施
  • (转载)hibernate缓存
  • ***检测工具之RKHunter AIDE
  • .360、.halo勒索病毒的最新威胁:如何恢复您的数据?
  • .java 9 找不到符号_java找不到符号
  • .MyFile@waifu.club.wis.mkp勒索病毒数据怎么处理|数据解密恢复
  • .NET Compact Framework 3.5 支持 WCF 的子集