我们先来看一下Android官网画的activity生命周期图:
为了更加详细解释它的生命周期,我特意写了一个demo通过打印日志来分析具体流程。Java源代码如下:
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);
Log.d(TAG, "onCreate");
}
@Override
protected void onRestart() {
super.onRestart();
Log.d(TAG, "onRestart");
}
@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");
}
public void startNewActivity(View view) {
Intent intent = new Intent(this, ActivityB.class);
startActivity(intent);
}
}
布局文件源码如下(Button的 android:onClick属性设置成了 startNewActivity,点击Button时会调用上面的 startNewActivity):
<android.support.constraint.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context="com.huawei.listviewdemo.MainActivity">
<Button
android:id="@+id/button"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:onClick="startNewActivity"
android:text="@string/button_text"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintLeft_toLeftOf="parent"
app:layout_constraintRight_toRightOf="parent"
app:layout_constraintTop_toTopOf="parent" />
</android.support.constraint.ConstraintLayout>
第一种情况:也就是首次启动这个activity,比如从桌面点击这个应用程序,日志如下:
10-18 13:30:37.834 5370 5370 D MainActivity: onCreate
10-18 13:30:37.836 5370 5370 D MainActivity: onStart
10-18 13:30:37.841 5370 5370 D MainActivity: onResume
生命周期顺序是 onCreate -> onStart -> onResume. 这个是最简单的,想必大家都懂。
第二种情况:也就是图中的 “Another activity comes into the foreground”,在这里我们点击Button启动第二个activity,让第二个activity进入前台,日志如下:
10-18 14:30:25.941 5370 5370 D MainActivity: onPause
10-18 14:30:26.249 5370 5370 D MainActivity: onStop
由于第二个activity进入了前台,MainActivity就不可见了,所以满足“The activity is no longer visible”,所以调用了 onStop方法。
第三种情况:我们接着第二种情况往下运行,此时第二个activity位于前台,我们按返回键,也就是满足图中的“User navigates to the activity”,日志如下:
10-18 14:40:48.476 5370 5370 D MainActivity: onRestart
10-18 14:40:48.477 5370 5370 D MainActivity: onStart
10-18 14:40:48.478 5370 5370 D MainActivity: onResume
接着第二步的 onStop方法,调用顺序是 onRestart -> onStart -> onResume。
第四种情况:我们按下HOME键,看下日志情况:
10-18 14:45:18.976 5370 5370 D MainActivity: onPause
10-18 14:45:19.228 5370 5370 D MainActivity: onStop
由于按了HOME键,该activity也是不可见的,所以满足“The activity is no longer visible”,所以调用顺序是 onPause -> onStop。
第五种情况:我们进入近期任务栏,删除刚才的程序,也就是模拟图中的“App process killed”,应用进程被杀死,然后再从桌面点击进入(User navigates to the activity),日志如下:
10-18 14:52:44.489 6353 6353 D MainActivity: onCreate
10-18 14:52:44.491 6353 6353 D MainActivity: onStart
10-18 14:52:44.496 6353 6353 D MainActivity: onResume
可以看到,进程被杀死后,调用顺序是 onCreate -> onStart -> onResume,跟图中的顺序相符。
第六种情况:我们把点击Button的实现改成启动一个Dialog样式的Activity,这样的话,MainActivity虽然在后台,但是可见,我们把AndroidManifest.xml 改成如下这个样子:
<activity android:name=".ActivityB"
android:theme="@style/Theme.AppCompat.Dialog"></activity>
startNewActivity的实现仍然不变:
public void startNewActivity(View view) {
Intent intent = new Intent(this, ActivityB.class);
startActivity(intent);
}
点击Button,日志如下:
10-18 16:01:58.344 7494 7494 D MainActivity: onPause
再点击下这个Dialog周围让其消失,此时日志如下:
10-18 16:02:58.126 7494 7494 D MainActivity: onResume
由此说明,当后台activity可见时,它的 onPause被调用,再次处于前台时,onResume被调用。
第七种情况:我们一直按返回键,直到回到桌面,日志如下:
10-18 16:04:46.132 7494 7494 D MainActivity: onPause
10-18 16:04:46.382 7494 7494 D MainActivity: onStop
10-18 16:04:46.383 7494 7494 D MainActivity: onDestroy
可以看到,调用顺序是 onPause -> onStop -> onDestroy。因为按返回键的话activity被彻底销毁了。
好了,看完这篇文章后,你觉得activity的生命周期还很难吗?面试问起的时候还用担心吗?大家可以自己照着这个思路,自己敲一遍代码运行看下效果,再加深下记忆。