最近在学习四大组件源码,对学习滴滴Virual APK有一定的帮助,自然就涉及到onNewIntent调用时机的一些基础知识,我在Activity 中重写了所有生命周期函数,加入了 Log。代码如下所示:
···
private String TAG;
protected String getTAG() {
return TAG;
}
protected void setTAG() {
this.TAG = this.getClass().getSimpleName();
}
@Override
protected void onCreate(Bundle savedInstanceState) {
setTAG();
LogUtil.d(getTAG(), "--onCreate--");
super.onCreate(savedInstanceState);
setContentView(getLayoutResId());
ButterKnife.bind(this);
getBundleData();
initViews();
initData();
}
@Override
protected void onNewIntent(Intent intent) {
LogUtil.d(getTAG(), "--onNewIntent--");
super.onNewIntent(intent);
}
@Override
protected void onRestart() {
LogUtil.d(getTAG(), "--onRestart--");
super.onRestart();
}
@Override
protected void onStart() {
LogUtil.d(getTAG(), "--onStart--");
super.onStart();
}
@Override
protected void onResume() {
LogUtil.d(getTAG(), "--onResume--");
super.onResume();
}
@Override
protected void onPause() {
LogUtil.d(getTAG(), "--onPause--");
super.onPause();
}
@Override
protected void onStop() {
LogUtil.d(getTAG(), "--onStop--");
super.onStop();
}
@Override
protected void onDestroy() {
LogUtil.d(getTAG(), "--onDestroy--");
super.onDestroy();
}
···
调试后整理 Activity 的 onNewIntent 调用时机如下图所示。
当 activity (假设为 A) 的 launchMode 为 singleTop 且 A 的实例已经在 task 栈顶,或者 launchMode 为 singleTask 且 A 的实例已在 task 栈里 (无论是栈顶还是栈中),再次启动 activity A 时,便不会调用 onCreate() 去产生新的实例,而是调用 onNewIntent() 并重用 task 栈里的 A 实例。
如果 A 在栈顶,那么调用顺序依次是 A.onPause() –> A.onNewIntent() –> A.onResume()。A 的 launchMode 可以是 singleTop 或者是 singlTask.
如果 A 不在栈顶,此时它处于 A.onStop() 状态,当再次启动时,调用顺序依次是 [A.onStop()] –> A.onNewIntent() –> A.onRestart() –> A.onStart() –> A.onResume()。A 的 launchMode 只能是 singleTask。google 到的其它大多文章描述的是这种情况。
另外,网上的文章在谈及 activity 的生命周期时,往往只说明单个 activity 的生命周期,而不说明从一个 activity 进入到另一个 activity 时,或者从一个 activity 返回到上一个 activity 时这些函数的调用顺序。现整理如下图所示:
可见原则是优先把目标 activity 尽快展示出来,等目标 activity 展现出来后,再在后台执行自身的 onStop,或者以及 onDestroy。而并不是先执行完自己的 onStop/onDestroy 再去执行目标 activity 的 onCreate/onRestart,最后留一个问题,大家知道为什么这样设计吗?