前言
Activity的生命周期都应该熟悉,Fragment其标准生命周期也仅比Activity多出一些流程,如onCreateView(),比较好理解。了解Activity、Fragment的基础知识可以先转头去看下我的博文:
认识ViewPager页面加载机制
ViewPager的页面缓存机制默认是两个,如果有三个Fragment,实际上三个都加载了。它是以当前页开始计算,向前缓存一个,向后缓存一个,即默认为两个,而显示给用户看的当前页面已经加载出来了。所以说如果有三个Fragment,默认加载值是2,实际上三个都已经加载了。 这波能理解就好,不慌,问题不大。
ViewPager.setOffscreenPageLimit(2);
解决生命周期只走一次的方案
方案一:
- 设置Viewpager的缓存机制,不缓存除当前页以外的页面数据,所见即所得,离开即销毁;
结果:此方案对需求改动较大,且较影响用户体验; 所以答案是不合适!
方案二 :
- 重载Fragment.onHiddenChanged(boolean hidden)方法,其参数hidden代表当前fragment显隐状态改变时,是否为隐藏状态,可通过check此参数作处理;
结果:此方案局限在于本方法的系统调用时间发生在显隐状态改变时,但第一次显示时此方法并不调用;
方案三:
- 重载Fragment.setUserVisibleHint(boolean isVisibleToUser)方法,其参数isVisibleToUser顾名思义最接近我们的需求,代表页面是否“真正”对使用者显示;
结果:局限在于此方法的第一次系统调用甚至早于Fragment的onCreate方法,故其第一次调用时isVisibleToUser值总为false,影响我们对生命周期顺序的判定;
-
Fragment1 - isVisibleToUser - false //(多余)
-
Fragment1 - isVisibleToUser - true
-
Fragment1 - isVisibleToUser - false
-
Fragment2 - isVisibleToUser - false //(多余)
-
Fragment2 - isVisibleToUser - true
-
Fragment2 - isVisibleToUser - false
最后的抉择——第三个方案进行改进
//切换刷新 protected boolean isCreated = false;
@Override public void onCreate(@Nullable Bundle savedInstanceState) { super.onCreate(savedInstanceState); isCreated = true; } @Override public void setUserVisibleHint(boolean isVisibleToUser) { super.setUserVisibleHint(isVisibleToUser); if (!isCreated) { return; } //刷新画面 if (isVisibleToUser) { initData(); } }
小结:
此标记请勿放在Fragment的onActivtyCreate方法中,因为此方法生命周期落后于setUserVisibleHint的判断。