天天看点

Activity详解(生命周期、以各种方式启动Activity、状态保存,完全退出等)

    简单的说:activity就是布满整个窗口或者悬浮于其他窗口上的交互界面。在一个应用程序中通常由多个activity构成,都会在manifest.xml中指定一个主的activity,如下设置

<actionandroid:name="android.intent.action.main" />

当程序第一次运行时用户就会看这个activity,这个activity可以通过启动其他的activity进行相关操作。当启动其他的activity时这个当前的这个activity将会停止,新的activity将会压入栈中,同时获取用户焦点,这时就可在这个activity上操作了。都知道栈是先进后出的原则,那么当用户按back键时,当前的这个activity销毁,前一个activity重新恢复。

先看下图:

Activity详解(生命周期、以各种方式启动Activity、状态保存,完全退出等)

这个图不再多说什么,下面我们通过一个实例来说明问题。新建工程,编写如下代码:

[java] view

plain copy

 print?

<span style="font-size:18px;">package com.android.ttx.actiitylifedemo;  

import android.app.activity;  

import android.os.bundle;  

import android.util.log;  

import android.view.keyevent;  

public class activitylifedemo extends activity {  

    private final static string tag="activitylifedemo";  

    @override  

    public void oncreate(bundle savedinstancestate) {  

        super.oncreate(savedinstancestate);  

        setcontentview(r.layout.main);  

        log.i(tag, "oncreate");  

    }  

    protected void onstart() {  

        log.i(tag, "onstart");  

        super.onstart();  

    protected void onrestart() {  

        log.i(tag, "onrestart");  

        super.onrestart();  

    protected void onresume() {  

        log.i(tag, "onresume");  

        super.onresume();  

    protected void onpause() {  

        log.i(tag, "onpause");  

        super.onpause();  

    protected void onstop() {  

        log.i(tag, "onstop");  

        super.onstop();  

    protected void ondestroy() {  

        log.i(tag, "ondestroy");  

        super.ondestroy();  

}  

</span>  

代码很简单,只涉及到一个activity,一些用户的操作,我们通过记录操作和打印日志的方式来看看activity的生命周期过程。

1、  运行

看到如下打印日志:

08-31 08:46:53.916: info/activitylifedemo(312): oncreate

08-31 08:46:53.916: info/activitylifedemo(312): onstart

08-31 08:46:53.916: info/activitylifedemo(312): onresume

2、按下返回按键: 

08-31 09:29:57.396: info/activitylifedemo(354): onpause

08-31 09:29:58.216: info/activitylifedemo(354): onstop

08-31 09:29:58.216: info/activitylifedemo(354): ondestroy

3、长按home键,弹出最近打开过的应用程序,点击activitylifedemo

08-31 08:51:46.916: info/activitylifedemo(312): oncreate

08-31 08:51:46.916: info/activitylifedemo(312): onstart

08-31 08:51:46.936: info/activitylifedemo(312): onresume

4、按home键

08-31 08:53:32.676: info/activitylifedemo(312): onpause

08-31 08:53:33.796: info/activitylifedemo(312): onstop

5、在alllist中点击打开

08-31 08:54:14.286: info/activitylifedemo(312): onrestart

08-31 08:54:14.286: info/activitylifedemo(312): onstart

08-31 08:54:14.296: info/activitylifedemo(312): onresume

通过日志信息,我们可以看到。activity的启动过程:oncreate—onstart—onresume;下返回键时:onpause—onstop—ondestroy 正如上面说是,当按下返回键时,此activity弹出栈,程序销毁。确实如此,我们再次 打开时的启动过程又回到oncreate—onstart—onresume。ok,启动之后按下home键,回到launcher,查看打印信息:onpause—onstop,再次打开的运行过程:onrestart—onstart—onresume。

我们通过对activity的各种操作,构成了activity的生命周期,我们看到无论对activity做如何的操作,都会接收到相关的回调方法,那么我们在开发的过程中通过这些回调方法就可以写工作,比如说释放一些重量级的对象,网络连接,数据库连接,文件读等等。

以下是各个方法的详细说明:

oncreate():当 activity 第一次创建时会被调用。在这个方法中你需要完成所有的正常静态设置 ,比如创建一个视图( view )、绑定列表的数据等等。如果能捕获到 activity 状态的话,这个方法传递进来的 bundle 对象将存放了 activity 当前的状态。调用该方法后一般会调用 onstart() 方法。

onrestart():在 activity 被停止后重新启动时会调用该方法。其后续会调用 onstart 方法。

onstart()à当 activity 对于用户可见前即调用这个方法。如果 activity回到前台则接着调用 onresume() ,如果 activity 隐藏则调用onstop()

onresume():在 activity 开始与用户交互前调用该方法。在这时该activity 处于 activity 栈的顶部,并且接受用户的输入。其后续会调用 onpause() 方法。

onpause():在系统准备开始恢复其它 activity 时会调用该方法。这个方法中通常用来提交一些还没保存的更改到持久数据 中,停止一些动画或其它一些耗 cpu 的操作等等。无论在该方法里面进行任何操作,都需要较快速完成,因为如果它不返回的话,下一个 activity 将无法恢复出来。如果 activity 返回到前台将会调用 onresume() ,如果 activity 变得对用户不可见了将会调用onstop() 。

onstop():在 activity 对用户不可见时将调用该方法。可能会因为当前 activity 正在被销毁,或另一个 activity (已经存在的activity 或新的 activity )已经恢复了正准备覆盖它,而调用该方法。如果 activity 正准备返回与用户交互时后续会调用onrestart ,如果 activity 正在被释放则会调用 ondestroy 。

ondestroy():在 activity 被销毁前会调用该方法。这是 activity 能接收到的最后一个调用。可能会因为有人调用了 finish 方法使得当前activity 正在关闭,或系统为了保护内存临时释放这个 activity的实例,而调用该方法。你可以用 isfinishing 方法来区分这两种不同的情况。

要启动一个新的activity,我们可以通过调用context中的startactivity来启动。像这样:

<span style="font-size:18px;">intent intent = new intent(this, activitydemo.class);  

startactivity(intent);  // activitydemo是需要启动的activity类  

通过上面的方法可以启动新的activity了,但如果我要从当前的activity中传递数据到新的activity呢?很简单:

<span style="font-size:18px;">intent intent = new intent(this,activitydemo.class);  

bundle bundle = new bundle();  

bundle.putboolean("bool_key", true);  

intent.putextras(bundle);  

startactivity(intent);  

还有,有时候我们需要启动带返回值的activity,简单的说就是需要新启动的activity返回时将值传递给启动它的activity,像这样:

<span style="font-size:18px;">intent intent = new intent(activitylifedemo.this,revalueactivity.class);  

startactivityforresult(intent, 0x1001);  

activitylifedemo是当前的activity,启动revalueactivity,我们在activitylifedemo中需要获取revalueactivity传回来的值。那么在revalueactivity中就必须这样写:

<span style="font-size:18px;">intent intent  = new intent();  

intent.putextra("revalue_key","haha-revalueactivity");  

setresult(0x1001, intent);</span>  

那么“revalue_key”值在哪里获取呢?必须重写onactivityresult方法,通过判断requestcode,来确定

<span style="font-size:18px;">if(requestcode==0x1001){  

            string str = data.getstringextra("revalue_key");  

            log.i(tag, "返回的值为:"+str);  

        }  

好了,详细的请看代码吧。下载地址:http://download.csdn.net/detail/tangcheng_ok/3580700

通过重写onsaveinstancestate()方法来实现activity的运行状态,请注意以下几点:

1)由于activity 对象被暂停或停止时,它仍然保留在内存里面,关于它的成员信息和当前状态都是活动的,所以此时可以保存activity的状态,从而使用户所作的activity的更改保存在内存中

2)  当系统回收内存而将activity销毁时,就无法保存其状态,所以需要调用onsaveinstancestate()方法来实现状态的保存

3)  很多情况并不需要保持状态信息,比如按下返回键直接关闭程序,所以并不能保证会调用onsaveinstancestate。如果调用了该方法,一般是在onstop 方法之前且可能在 onpause 之后调用。尽管如此,即使你没做任何操作或没有实现 onsaveinstancestate() 方法,你的 activity 状态也能通过activity 类里面默认实现的 onsaveinstancestate 方法恢复出来。特别是会为布局中的视图( view )默认调用onsaveinstancestate 方法,并在这个方法中允许每一个视图提供它需要恢复的任何信息。几乎每一个 android框架中的 widget 都视情况实现了这个方法。

注:因为 onsaveinstancestate 方法不一定会被调用,所以你应该只是用它来保存一些 activity 的转换过程状态(即 ui 的状态),而不能用来保存永久性数据。但你可以用 onpause 方法在用户离开 activity 时来保存永久性数据,比如需要保存到数据库的数据。

有一个很好的方法可以用来检验应用程序保存状态的能力,就是简单地旋转你的设备来改变屏幕的方向。因为当屏幕方向改变时,系统为了给新的方向提供一个可能合适的代替资源,会销毁 activity 并新建一个新的。由于这个原因,你的 activity 是否能在其重新创建时完成保存状态就显得尤为重要,因为用户经常会在使用应用程序时旋转屏幕的。

上文部分内容参考了:http://blog.csdn.net/sam_zhang1984/article/details/6430817,在这里多谢这位博主了。

通过上面的介绍,我们知道当点击back键时,程序调用了ondestroy方法,程序退出了,但是我们查看其进程,发现调用了ondestroy方法之后这个activity还在运行。甚至调用了finish()方法之后程序还能在进程中看到。通过下面这种方式可以实现程序的完全退出:

<span style="font-size:18px;">intent intent = new intent();  

intent.setclass(context,mainactivity.class);  

intent.setflags(intent.flag_activity_clear_top);  

intent.putextra(“flag”,exit_application);  

context.startactivity(intnet);   

ok,到此为止。该准备下一篇了。