Android开发学习之路--Activity之生命周期

来源:互联网 发布:我的世界编程网站 编辑:程序博客网 时间:2024/05/01 05:46

    其实这篇文章应该要在介绍Activity的时候写的,不过那个时候还不怎么熟悉Activity,还是在这里详细介绍下好了。还是参考下官方文档的图吧:


    从上面的流程,我们可以看出首先就是打开APP,开始执行到当前的FirstActivity了,接着anroid系统会调用onCreate方法,然后是onStart方法,onResume方法,之后FirstActivity就完全启动好了,展现到前台,如果这个时候用intent启动了另一个SecondActivity的话,那么就会调用FirstActivity的onPause方法,通过back键返回到FirstActivity就会调用onResume方法,而此时SecondActivity会销毁会调用onStop方法,如果这个时候又从FirstActivity调用到SecondActivity,那么会调用onRestart方法,接着onStart方法,然后循环。但是如果这个时候内存不够,或者finish了Activity,那么就会调用onDestory方法,Activity就结束了,如果启动就必须重新创建了。

    根据官方文档,介绍了Activiy的三个重要的循环对:

    1、整个生命周期:从第一个onCreate方法到onDestroy方法。Activity需要在onCreate中准备所有的全局状态,需要在onDestroy方法中释放所有的资源。比如,我们有一个线程在后台跑需要从网络下载数据,我们再onCreate方法里创建线程,在onDestroy方法里停止线程。

    2、可见生命周期:从onStart方法到onStop方法。顾名思义叫做可见生命周期,其实是因为这个过程用户可以很直观的在屏幕上看到Activity,在这两个过程之间我们可以保持我们需要展示给用户的资源。比如,我们可以在onStart方法中注册一个BroadcastReceiver监视一些改变来设置UI,当我们不在看到显示的UI的时候,我们需要在onStop方法中取消监视。onStart方法和onStop方法可以在用户可见和隐藏不多切换的时候被多次调用。

    3、前台生命周期:这个过程从onResume方法到onPause方法。这个过程Activity是聚焦到最上层展现给用户。Activity从onResume到onPause可能切换非常频繁-例如当设备处于休眠状态,当result和一个新的intent被发送后,所以这里的代码需要非常简洁。

public class Activity extends ApplicationContext {     protected void onCreate(Bundle savedInstanceState);     protected void onStart();          protected void onRestart();     protected void onResume();     protected void onPause();     protected void onStop();     protected void onDestroy(); }
    关于各个函数在这里也做个简单的介绍,其实也是参考的官方文档:

    1、onCreate方法:当Activity第一次被创建,调用onCreate方法。这里我们需要完成初始化操作:加载view布局,绑定事件等。

    2、onRestart方法:在Activity被停止后调用该方法重新启动。

    3、onStart方法:当Activity将要展示给用户的时候被调用。

    4、onResume方法:当Activity可以和用户交互的时候被调用。这个时候Activity在Activity栈的栈顶,等待用户交互。

    5、onPause方法:当系统准备启动或恢复之前的Activity的时候被调用,这个时候,我们需要保存数据,停止动画和其他一些耗费CPU资源的事情。这个实现必须非常快,因为如果函数未返回,会影响下一个Activity的resume。

    6、onStop方法:当Activity长时间不可见的时候被调用。

    7、onDestroy方法:当Activity被销毁时我们会收到最后一个调用。不管是调用finish方法还是因为系统没有为了释放空间都会调用该方法。

    这些函数也讲解了,那么我们来了解下了上述的Activity栈,听名字我们也可以明白什么意思了,就是一个一个Activity就像栈一样,栈的话其实就和我们以前用过的储蓄硬币的小罐子,一个一个压进去,然后之后栈顶的取出来之后栈底的才能取出来,Activity也是如此。比如我们运行了FirstActivity,那么FirstActivity就进栈了,接着调用intent运行了SecondActivity,那么SecondActivity进栈了,FirstActiviy在栈底,接着按back键,SecondActivity出栈被销毁,FirstActiviy又出来了在栈顶。所有的显示在我们面前的Activity就是在栈顶的Activity了。

    讲了这么多,接下来还是通过实例来看看运行结果吧。这里我们通过logcat来查看调用的情况,ui显示看信息已经不能满足要求了,下面我们新建个工程ActivityLifetime,具体新建就按照前几篇文章来,这里不多加赘述。

    这里需要为了实现7种方法都有调用,所以就用了三个Activity,第一个是主的Activity,第二个是普通的Activity,第三个是Dialog。首先第一个Activity添加两个按钮,如下所示


    新建TestActivity和TestDialog两个Activity。代码基本相同,都加上textview。只是显示内容不一样,如下图:

    

     但是这样仅仅是显示不一样的内容,不是dialog,那么需要在TestDialog的AndroidManifest中添加主题了:

<activity android:name=".TestDialog" android:theme="@style/Theme.AppCompat.Dialog"></activity>
     既然这些都完成了,那么剩下就是把两个button按钮的功能给实现  

protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_main);        startTestActivity = (Button)findViewById(R.id.button1);        startTestDialog = (Button)findViewById(R.id.button2);        startTestActivity.setOnClickListener(new View.OnClickListener() {            @Override            public void onClick(View view) {                Intent intentActivity = new Intent(MainActivity.this, TestActivity.class);                startActivity(intentActivity);            }        });        startTestDialog.setOnClickListener(new View.OnClickListener() {            @Override            public void onClick(View view) {                Intent intentDialog = new Intent(MainActivity.this, TestDialog.class);                startActivity(intentDialog);            }        });    }

    基本代码写好了,好像忘了我们的目的是生命周期,那就用logcat来看看打印信息吧。在三个Activity中添加7个方法的log信息。

    

package com.example.jared.activitylifetime;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;public class MainActivity extends AppCompatActivity {    public static  String TAG = "MainActivity";    private Button startTestActivity;    private Button startTestDialog;    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        Log.d(TAG, "onCreate is Called");        setContentView(R.layout.activity_main);        startTestActivity = (Button)findViewById(R.id.button1);        startTestDialog = (Button)findViewById(R.id.button2);        startTestActivity.setOnClickListener(new View.OnClickListener() {            @Override            public void onClick(View view) {                Intent intentActivity = new Intent(MainActivity.this, TestActivity.class);                startActivity(intentActivity);            }        });        startTestDialog.setOnClickListener(new View.OnClickListener() {            @Override            public void onClick(View view) {                Intent intentDialog = new Intent(MainActivity.this, TestDialog.class);                startActivity(intentDialog);            }        });    }    @Override    protected void onStart() {        super.onStart();        Log.d(TAG, "onStart is Called");    }    @Override    protected void onResume() {        super.onResume();        Log.d(TAG, "onResume is Called");    }    @Override    protected void onPause() {        super.onPause();        Log.d(TAG, "onPause is Called");    }    @Override    protected void onStop() {        super.onStop();        Log.d(TAG, "onStop is Called");    }    @Override    protected void onDestroy() {        super.onDestroy();        Log.d(TAG, "onDestroy is Called");    }    @Override    protected void onRestart() {        super.onRestart();        Log.d(TAG, "onRestart is Called");    }}

    好了,代码已经全部添加完毕,那么我们来看看结果是不是如上面所说的一样呢?首先运行打开app,查看logcat如下:

02-03 07:38:59.105 6967-6967/com.example.jared.activitylifetime D/MainActivity: onCreate is Called02-03 07:38:59.485 6967-6967/com.example.jared.activitylifetime D/MainActivity: onStart is Called02-03 07:38:59.485 6967-6967/com.example.jared.activitylifetime D/MainActivity: onResume is Called

    可以看出结果和我们所学的一样,接着打开另一个Activity,查看logcat如下:


02-03 07:45:32.425 6967-6967/com.example.jared.activitylifetime D/MainActivity: onPause is Called02-03 07:45:36.305 6967-6967/com.example.jared.activitylifetime D/MainActivity: onStop is Called
    接着按back键回退到MainActivity:

02-03 07:45:42.875 6967-6967/com.example.jared.activitylifetime D/MainActivity: onRestart is Called02-03 07:45:42.875 6967-6967/com.example.jared.activitylifetime D/MainActivity: onStart is Called02-03 07:45:42.875 6967-6967/com.example.jared.activitylifetime D/MainActivity: onResume is Called
    从上可以知道首先app开启后启动了MainActivity,然后调用了onCreate->onStart->onResume,然后显示了MainActivity的UI,接着打开TestActivity的时候调用了onPause->onStop来停止MainActivity,显示TestActivity的UI,当按back键后,TestActivity被销毁,然后MainActivity调用了onRestart->onStart->onResume来重新显示MainActivity的UI。

   接着我们看看打开Dialog的效果,查看logcat如下:


02-03 08:09:14.975 6967-6967/com.example.jared.activitylifetime D/MainActivity: onPause is Called

    然后按back键退回到MainActivity:

02-03 08:10:02.335 6967-6967/com.example.jared.activitylifetime D/MainActivity: onResume is Called

    因为MainActivity还是显示给用户的,只是在他之上还有一个dialog,所以不会stop Activity,而仅仅是Pause,然后Resume。

    还有一点需要注意的就是如果在MainActivity之后启动了TestActivity,然后由于系统内存不够了,然后MainActivity会被Destroy,释放,虽然当我们再次按back键会重新创建MainActivity给人的感觉基本没事,但是万一MainActivity中原本就有用户输入的数据,比如注册的时候填写了一些信息,发现错了,想回去重新填写,那么就会出问题。

    Android肯定想到了这个问题,有一个函数onSaveInstanceState方法就可以实现这个功能,在系统kill该MainActivity之前会先调用这个方法。然后我们可以在这个方法中保存一些信息。如下:

 protected void onSaveInstanceState(Bundle outState) {        super.onSaveInstanceState(outState);        String SaveData = "We must save the info";        String key = "SaveInfo";        outState.putString(key, SaveData);    }
     信息保存好后,那么什么时候读取呢?之前那7个方法,也就onCreate方法有传入的参数了。那也是个Bundle,那个Bundle的话一般是null,只有这个会系统回收,需要重新create的时候才调用。

protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        Log.d(TAG, "onCreate is Called");        setContentView(R.layout.activity_main);        if(savedInstanceState != null) {            String key = "SaveInfo";            String savedData = savedInstanceState.getString(key);            Log.d(TAG, savedData);        }}
    Activity的生命周期基本是就学到这里了。

    


2 0