zoukankan      html  css  js  c++  java
  • [转]: 两分钟彻底让你明白Android Activity生命周期(图文)!

    转自:http://blog.csdn.net/android_tutor/article/details/5772285

    大家好,今天给大家详解一下Android中Activity的生命周期,我在前面也曾经讲过这方面的内容,但是像网上大多数文章一样,基本都是翻译Android API,过于笼统,相信大家看了,会有一点点的帮助 ,但是还不能完全吃透,所以我今天特意在重新总结一下.

    首先看一下Android api中所提供的Activity生命周期图(不明白的,可以看完整篇文章,在回头看一下这个图,你会明白的):

    Activity其实是继承了ApplicationContext这个类,我们可以重写以下方法,如下代码:

    [java] view plaincopy
     
    1. public class Activity extends ApplicationContext {  
    2.        protected void onCreate(Bundle savedInstanceState);  
    3.          
    4.        protected void onStart();     
    5.          
    6.        protected void onRestart();  
    7.          
    8.        protected void onResume();  
    9.          
    10.        protected void onPause();  
    11.          
    12.        protected void onStop();  
    13.          
    14.        protected void onDestroy();  
    15.    }  

    为了便于大家更好的理解,我简单的写了一个Demo,不明白Activity周期的朋友们,可以亲手实践一下,大家按照我的步骤来。

    第一步:新建一个Android工程,我这里命名为ActivityDemo.

    第二步:修改ActivityDemo.java(我这里重新写了以上的七种方法,主要用Log打印),代码如下:

    [java] view plaincopy
     
    1. package com.tutor.activitydemo;  
    2. import android.app.Activity;  
    3. import android.os.Bundle;  
    4. import android.util.Log;  
    5. public class ActivityDemo extends Activity {  
    6.      
    7.     private static final String TAG = "ActivityDemo";  
    8.       
    9.     public void onCreate(Bundle savedInstanceState) {  
    10.         super.onCreate(savedInstanceState);  
    11.         setContentView(R.layout.main);  
    12.           
    13.         Log.e(TAG, "start onCreate~~~");  
    14.     }  
    15.       
    16.     @Override  
    17.     protected void onStart() {  
    18.         super.onStart();  
    19.         Log.e(TAG, "start onStart~~~");  
    20.     }  
    21.       
    22.     @Override  
    23.     protected void onRestart() {  
    24.         super.onRestart();  
    25.         Log.e(TAG, "start onRestart~~~");  
    26.     }  
    27.       
    28.     @Override  
    29.     protected void onResume() {  
    30.         super.onResume();  
    31.         Log.e(TAG, "start onResume~~~");  
    32.     }  
    33.       
    34.     @Override  
    35.     protected void onPause() {  
    36.         super.onPause();  
    37.         Log.e(TAG, "start onPause~~~");  
    38.     }  
    39.       
    40.     @Override  
    41.     protected void onStop() {  
    42.         super.onStop();  
    43.         Log.e(TAG, "start onStop~~~");  
    44.     }  
    45.       
    46.     @Override  
    47.     protected void onDestroy() {  
    48.         super.onDestroy();  
    49.         Log.e(TAG, "start onDestroy~~~");  
    50.     }  
    51.       
    52. }  

    第三步:运行上述工程,效果图如下(没什么特别的):

    核心在Logcat视窗里,如果你还不会用Logcat你可以看一下我的这篇文章 Log图文详解(Log.v,Log.d,Log.i,Log.w,Log.e) ,我们打开应用时先后执行了onCreate()->onStart()->onResume三个方法,看一下LogCat视窗如下:

    BACK键:

    当我们按BACK键时,我们这个应用程序将结束,这时候我们将先后调用onPause()->onStop()->onDestory()三个方法,如下图所示:

    HOME键:

    当我们打开应用程序时,比如浏览器,我正在浏览NBA新闻,看到一半时,我突然想听歌,这时候我们会选择按HOME键,然后去打开音乐应用程序,而当我们按HOME的时候,Activity先后执行了onPause()->onStop()这两个方法,这时候应用程序并没有销毁。如下图所示:

    而当我们再次启动ActivityDemo应用程序时,则先后分别执行了onRestart()->onStart()->onResume()三个方法,如下图所示:

    这里我们会引出一个问题,当我们按HOME键,然后再进入ActivityDemo应用时,我们的应用的状态应该是和按HOME键之前的状态是一样的,同样为了方便理解,在这里我将ActivityDemo的代码作一些修改,就是增加一个EditText。

    第四步:修改main.xml布局文件(增加了一个EditText),代码如下:

    [java] view plaincopy
     
    1. <?xml version="1.0" encoding="utf-8"?>  
    2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  
    3.     android:orientation="vertical"  
    4.     android:layout_width="fill_parent"  
    5.     android:layout_height="fill_parent"  
    6.     >  
    7. <TextView    
    8.     android:layout_width="fill_parent"   
    9.     android:layout_height="wrap_content"   
    10.     android:text="@string/hello"  
    11.     />  
    12. <EditText  
    13.     android:id="@+id/editText"  
    14.     android:layout_width="fill_parent"  
    15.     android:layout_height="wrap_content"  
    16. />  
    17. </LinearLayout>  

    第五步:然后其他不变,运行ActivityDemo程序,在EditText里输入如"Frankie"字符串(如下图:)

    这时候,大家可以按一下HOME键,然后再次启动ActivityDemo应用程序,这时候EditText里并没有我们输入的"Frankie"字样,如下图:

    这显然不能称得一个合格的应用程序,所以我们需要在Activity几个方法里自己实现,如下第六步所示:

    第六步修改ActivityDemo.java代码如下:

    [java] view plaincopy
     
    1. package com.tutor.activitydemo;  
    2. import android.app.Activity;  
    3. import android.os.Bundle;  
    4. import android.util.Log;  
    5. import android.widget.EditText;  
    6. public class ActivityDemo extends Activity {  
    7.      
    8.     private static final String TAG = "ActivityDemo";  
    9.     private EditText mEditText;  
    10.     //定义一个String 类型用来存取我们EditText输入的值  
    11.     private String mString;  
    12.     public void onCreate(Bundle savedInstanceState) {  
    13.         super.onCreate(savedInstanceState);  
    14.         setContentView(R.layout.main);  
    15.         mEditText = (EditText)findViewById(R.id.editText);  
    16.         Log.e(TAG, "start onCreate~~~");  
    17.     }  
    18.       
    19.     @Override  
    20.     protected void onStart() {  
    21.         super.onStart();  
    22.         Log.e(TAG, "start onStart~~~");  
    23.     }  
    24.     //当按HOME键时,然后再次启动应用时,我们要恢复先前状态  
    25.     @Override  
    26.     protected void onRestart() {  
    27.         super.onRestart();  
    28.         mEditText.setText(mString);  
    29.         Log.e(TAG, "start onRestart~~~");  
    30.     }  
    31.       
    32.     @Override  
    33.     protected void onResume() {  
    34.         super.onResume();  
    35.         Log.e(TAG, "start onResume~~~");  
    36.     }  
    37.       
    38.     //当我们按HOME键时,我在onPause方法里,将输入的值赋给mString  
    39.     @Override  
    40.     protected void onPause() {  
    41.         super.onPause();  
    42.         mString = mEditText.getText().toString();  
    43.         Log.e(TAG, "start onPause~~~");  
    44.     }  
    45.       
    46.     @Override  
    47.     protected void onStop() {  
    48.         super.onStop();  
    49.         Log.e(TAG, "start onStop~~~");  
    50.     }  
    51.       
    52.     @Override  
    53.     protected void onDestroy() {  
    54.         super.onDestroy();  
    55.         Log.e(TAG, "start onDestroy~~~");  
    56.     }  
    57.       
    58. }  

    第七步:重新运行ActivityDemo程序,重复第五步操作,当我们按HOME键时,再次启动应用程序时,EditText里有上次输入的"Frankie"字样,如下图如示:

    OK,大功基本告成,这时候大家可以在回上面看一下Activity生命周期图,我想大家应该完全了解了Activity的生命周期了,不知道你了解了没?

    ==以下内容来源:======

    http://blog.chinaunix.net/uid-24410388-id-3855498.html

    三个循环
    提供两个关于Activity的生命周期模型图示帮助理解:


     图1

    图2
    从图2所示的Activity生命周期不难看出,在这个图中包含了两层循环,

    第一层循环是onPause -> onResume -> onPause,

    第二层循环是onStop -> onRestart -> onStart -> onResume -> onPause -> onStop。我们可以将这两层循环看成是整合Activity生命周期中的子生命周期。第一层循环称为焦点生命周期,第二层循环称为可视生命周期。也就是说,第一层循环在Activity焦点的获得与失去的过程中循环,在这一过程中,Activity始终是可见的。而第二层循环是在Activity可见与不可见的过程中循环,在这个过程中伴随着Activity的焦点的获得与失去。也就是说,Activity首先会被显示,然后会获得焦点,接着失去焦点,最后由于弹出其他的Activity,使当前的Activity变成不可见。因此,Activity有如下3种生命周期:

    整体生命周期:onCreate -> ... ... -> onDestroy。
    可视生命周期:onStop -> ... ... -> onPause。
    焦点生命周期:onPause -> onResume。

    四个阶段
    上面7个生命周期方法分别在4个阶段按着一定的顺序进行调用,这4个阶段如下:

     开始Activity:在这个阶段依次执行3个生命周期方法:onCreate、onStart和onResume。
     Activity失去焦点:如果在Activity获得焦点的情况下进入其他的Activity或应用程序,这时当前的Activity会失去焦点。在这一阶段,会依次执行onPause和onStop方法。
    Activity重新获得焦点:如果Activity重新获得焦点,会依次执行3个生命周期方法:onRestart、onStart和onResume。
     关闭Activity:当Activity被关闭时系统会依次执行3个生命周期方法:onPause、onStop和onDestroy。
    如果在这4个阶段执行生命周期方法的过程中不发生状态的改变,那么系统会按着上面的描述依次执行这4个阶段中的生命周期方法,但如果在执行的过程中改变了状态,系统会按着更复杂的方式调用生命周期方法。

    在执行的过程中可以改变系统的执行轨迹的生命周期方法是onPause和onStop。如果在执行onPause方法的过程中Activity重新获得了焦点,然后又失去了焦点。系统将不会再执行onStop方法,而是按着如下的顺序执行相应的生命周期方法:

    onPause -> onResume-> onPause
    如果在执行onStop方法的过程中Activity重新获得了焦点,然后又失去了焦点。系统将不会执行onDestroy方法,而是按着如下的顺序执行相应的生命周期方法:

    onStop -> onRestart -> onStart -> onResume -> onPause -> onStop
     在图2所示的Activity生命周期里可以看出,系统在终止应用程序进程时会调用onPause、onStop和onDesktroy方法。而onPause方法排在了最前面,也就是说,Activity在失去焦点时就可能被终止进程,而onStop和onDestroy方法可能没有机会执行。因此,应该在onPause方法中保存当前Activity状态,这样才能保证在任何时候终止进程时都可以执行保存Activity状态的代码。
     
    七个方法 
    下面对其分别详细说明(文字中的粗体字表示后文中会经常用到的概念在第一次出现时会给出解释,之后后文不再详细说明): 

    1. void onCreate(Bundle savedInstanceState) 
    当Activity被第首次加载时执行。我们新启动一个程序的时候其主窗体的onCreate事件就会被执行。如果Activity被销毁后(onDestroy后),再重新加载进Task时,其onCreate事件也会被重新执行。注意这里的参数 savedInstanceState(Bundle类型是一个键值对集合,大家可以看成是.Net中的Dictionary)是一个很有用的设计,由于前面已经说到的手机应用的特殊性,一个Activity很可能被强制交换到后台(交换到后台就是指该窗体不再对用户可见,但实际上又还是存在于某个Task中的,比如一个新的Activity压入了当前的Task从而“遮盖”住了当前的 Activity,或者用户按了Home键回到桌面,又或者其他重要事件发生导致新的Activity出现在当前Activity之上,比如来电界面),而如果此后用户在一段时间内没有重新查看该窗体(Android通过长按Home键可以选择最近运行的6个程序,或者用户直接再次点击程序的运行图标,如果窗体所在的Task和进程没有被系统销毁,则不用重新加载,直接重新显示Task顶部的Activity,这就称之为重新查看某个程序的窗体),该窗体连同其所在的 Task和Process则可能已经被系统自动销毁了,此时如果再次查看该窗体,则要重新执行 onCreate事件初始化窗体。而这个时候我们可能希望用户继续上次打开该窗体时的操作状态进行操作,而不是一切从头开始。例如用户在编辑短信时突然来电,接完电话后用户又去做了一些其他的事情,比如保存来电号码到联系人,而没有立即回到短信编辑界面,导致了短信编辑界面被销毁,当用户重新进入短信程序时他可能希望继续上次的编辑。这种情况我们就可以覆写Activity的void onSaveInstanceState(Bundle outState)事件,通过向outState中写入一些我们需要在窗体销毁前保存的状态或信息,这样在窗体重新执行onCreate的时候,则会通过 savedInstanceState将之前保存的信息传递进来,此时我们就可以有选择的利用这些信息来初始化窗体,而不是一切从头开始。 

    2. void onStart()   activity变为在屏幕上对用户可见时调用。
    onCreate事件之后执行。或者当前窗体被交换到后台后,在用户重新查看窗体前已经过去了一段时间,窗体已经执行了onStop事件,但是窗体和其所在进程并没有被销毁,用户再次重新查看窗体时会执行onRestart事件,之后会跳过onCreate事件,直接执行窗体的onStart事件。 

    3. void onResume()   activity开始与用户交互时调用(无论是启动还是重新启动一个活动,该方法总是被调用的)。
    onStart事件之后执行。或者当前窗体被交换到后台后,在用户重新查看窗体时,窗体还没有被销毁,也没有执行过onStop事件(窗体还继续存在于Task中),则会跳过窗体的onCreate和onStart事件,直接执行onResume事件。 

    4. void onPause()   activity被暂停或收回cpu和其他资源时调用,该方法用于保存活动状态的,也是保护现场,压栈吧!
    窗体被交换到后台时执行。 

    5. void onStop()    activity被停止并转为不可见阶段及后续的生命周期事件时调用。
    onPause事件之后执行。如果一段时间内用户还没有重新查看该窗体,则该窗体的onStop事件将会被执行;或者用户直接按了Back键,将该窗体从当前Task中移除,也会执行该窗体的onStop事件。 


    6. void onRestart()   重新启动activity时调用。该活动仍在栈中,而不是启动新的活动。
    onStop事件执行后,如果窗体和其所在的进程没有被系统销毁,此时用户又重新查看该窗体,则会执行窗体的onRestart事件,onRestart事件后会跳过窗体的onCreate事件直接执行onStart事件。 

    7. void onDestroy()   activity被完全从系统内存中移除时调用,该方法被调用可能是因为有人直接调用onFinish()方法或者系统决定停止该活动以释放资源!
    Activity被销毁的时候执行。在窗体的onStop事件之后,如果没有再次查看该窗体,Activity则会被销毁。 

    最后用一个实际的例子来说明Activity的各个生命周期。假设有一个程序由2个Activity A和B组成,A是这个程序的启动界面。当用户启动程序时,Process和默认的Task分别被创建,接着A被压入到当前的Task中,依次执行了 onCreate, onStart, onResume事件被呈现给了用户;此时用户选择A中的某个功能开启界面B,界面B被压入当前Task遮盖住了A,A的onPause事件执行,B的 onCreate, onStart, onResume事件执行,呈现了界面B给用户;用户在界面B操作完成后,使用Back键回到界面A,界面B不再可见,界面B的onPause, onStop, onDestroy执行,A的onResume事件被执行,呈现界面A给用户。此时突然来电,界面A的onPause事件被执行,电话接听界面被呈现给用户,用户接听完电话后,又按了Home键回到桌面,打开另一个程序“联系人”,添加了联系人信息又做了一些其他的操作,此时界面A不再可见,其 onStop事件被执行,但并没有被销毁。此后用户重新从菜单中点击了我们的程序,由于A和其所在的进程和Task并没有被销毁,A的onRestart 和onStart事件被执行,接着A的onResume事件被执行,A又被呈现给了用户。用户这次使用完后,按Back键返回到桌面,A的 onPause, onStop被执行,随后A的onDestroy被执行,由于当前Task中已经没有任何Activity,A所在的Process的重要程度被降到很低,很快A所在的Process被系统结束
  • 相关阅读:
    PTA天梯赛L2
    图论-最短路算法
    配置自己的sublime
    testNG常用用法总结
    testng.xml文件的配置
    阿里云服务器 ECS Jenkins 安装教程
    jenkins报错:Problem accessing /jenkins/. Reason: HTTP ERROR 404
    jenkins安装详细教程
    jenkins下载插件无插件显示+离线下载插件方法
    如何让junit的测试跑多次
  • 原文地址:https://www.cnblogs.com/Jessy/p/3666299.html
Copyright © 2011-2022 走看看