下拉通知栏时发生了什么
在某个APP中,发现下拉通知栏的时候,正在播放的视频会暂停,于是有点好奇这段操作是不是在生命周期中实现的。在网上众多关于Activity生命周期的讨论中,很多人认为onPause()
和onStop()
的区别就是“部分遮挡”和“全部遮挡”,那按照这个猜测来分析一下这个过程:
首先,通知栏下拉一点点,符合一般描述中“Activity被部分遮挡”——onPause()
然后,通知栏完全落下之后,“Activity被全部遮挡”——onStop()
于是自己写了一个实例来验证:
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.i(TAG, "onCreate");
}
@Override
protected void onStart() {
super.onStart();
Log.i(TAG, "onStart");
}
@Override
protected void onResume() {
super.onResume();
Log.i(TAG, "onResume");
}
@Override
protected void onPause() {
super.onPause();
Log.i(TAG, "onPause");
}
@Override
protected void onStop() {
super.onStop();
Log.i(TAG, "onStop");
}
@Override
protected void onDestroy() {
super.onDestroy();
Log.i(TAG, "onDestroy");
}
}
启动APP时,毫无疑问,调用了onCreate()
→onStart()
→onResume()
;
完全下拉通知栏,然后上拉通知栏,发现没有日志打印,说明下拉通知栏对Activity的生命周期没有影响。
其他的“部分遮挡”——AlertDialog、Toast
经过测试不难发现,在Activity中弹出AlertDialog、Toast时,Activity的onPause()
并没有调用;笔者还尝试在MIUI系统中唤醒小爱同学,发现onPause()
仍然没有被调用。
但是在以下特殊的情况下,onPause()
会被调用:
- 自定义dialog继承自Activity
- 新启动的Activity主题设置为
android:theme=@android:style/Theme.Dialog
重新理解onPause()
跑去看文档发现了如下信息:
Method | Description |
---|---|
onPause() | Called when the system is about to start resuming a previous activity. This is typically used to commit unsaved changes to persistent data, stop animations and other things that may be consuming CPU, etc. Implementations of this method must be very quick because the next activity will not be resumed until this method returns. Followed by either onResume() if the activity returns back to the front, or onStop() if it becomes invisible to the user. |
发现了onPause()
和Activity的奇妙联系,就不难理解之前为什么没有被调用的问题了。
查看AlertDialog和Toast的源码,可以发现它们显示的原理,都是通过WindowManager.addView()
来显示的。也就是说,AlertDialog和Toast可以看做是当前Activity的一部分View,当然也不会对Activity的生命周期构成影响。
因此,onPause()
是否调用的关键就是,是否有另一个Activity参与进来了。
而网上流传甚广的onPause()
和onStop()
调用中提到的“遮挡”,应该修正为“被Activity遮挡”
至于官方文档中提到的,onPause()
之后会调用onStop()
或者onResume()
,前者很好理解,一般的退出、新启动一个全屏Activity、锁屏、返回HOME等操作都是这种情况;至于后者,笔者能想到的情况就是,弹出部分遮挡的Activity类型的对话框,然后按返回键。