前言
最近我在关注ViewPager2
的使用,期间一直基于官方的Demo调试android-viewpager2,今天遇到一个奇葩的问题,捉摸了半天最终找到原因,原来是Demo中布局的问题,事后感觉有必要分享一下这个过程,一来可以巩固View测量的知识,二来希望大家能避开这个坑;
阅读指南
- 代码基于android-viewpager2,看官老爷最好能下载源码亲身体会;
入坑现场
为了观察Fragment
的生命周期,我事先在CardFragment
类中,对生命周期方法进行埋点Log;
异常发生的操作步骤:
横屏进入CardFragmentActivity
或者CardFragmentActivity
竖屏切到横屏,控制台瞬间打印多个Fragment的生命周期Log,场面让人惊呆;
CardFragmentActivity横屏下布局
控制台Log输出
由于Log太长,一屏根本截不完,反正就是很多个Fragment
经历了onCreate
->onDestory
的所有过程;
操作前,只有Fragment2
创建并显示,理论上旋转屏幕之后,只有Fragment2
销毁并重建,不会调用其他Fragment
;现在问题发生在了,旋转之后有一堆Fragment
创建并且销毁,最终保留的也只有Fragment2
,这肯定是个Bug,虽然发生在一行代码都没有改的官方Demo上;
初步原因MATCH_PARENT计算失效
ViewPager2
目前只支持ItemView
的布局参数是MATCH_PARENT
,就是填充父布局的效果;由于ViewPager2
是基于RecyclerView
,理论上每个ItemView
一定会是MATCH_PARENT
,控制一屏只加载一个Item
,但是一旦MATCH_PARENT
计算失效,那么ViewPager2
基本上就是RecyclerView
的效果,瞬间多个Fragment
是可以解释通的;
ViewPager2
测量流程
ViewPager2
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
//测量mRecyclerView
measureChild(mRecyclerView, widthMeasureSpec, heightMeasureSpec);
int width = mRecyclerView.getMeasuredWidth();
int height = mRecyclerView.getMeasuredHeight();
int childState = mRecyclerView.getMeasuredState();
//宽高计算
width += getPaddingLeft() + getPaddingRight();
height += getPaddingTop() + getPaddingBottom();
//宽高约束
width = Math.max(width, getSuggestedMinimumWidth());
height = Math.max(height, getSuggestedMinimumHeight());
//设置自身高度