关于fragment重叠现象相信很多人遇到过,那到底是什么原因导致的呢?本质上还是从“内存重启”这个说法讲起( Fragment系列之调用getActivity为null的问题),前文已经解释了,这里就不重复概述了。
造成原因
1.重复replace|add Fragment
当应用重启发生后,我们在其onCreate()的方法中没有判断saveInstanceState==null,结果重新执行了一遍代码生成新的fragment对象,这和我们应用重启后从Bundle 中取出fragment造成了重复加载(replace情况下,如果没有加入回退栈,则不判断也不会造成重叠,但建议还是统一判断下)
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
// 在页面重启时,Fragment会被保存恢复,而此时再加载Fragment会重复加载,导致重叠 ;
if(saveInstanceState == null){
// 这里replace或add Fragment
}
}
2.使用show , hide控制Fragment
而在使用add方式进行加载fragment的时候,发生”内存重启”的时候,由于fragment的bug导致没有保存mHidden这个字段,所以界面重新加载的时候, 默认mHidden=false,所有的fragment都会是show状态显示出来(即VISIBLE),从而造成了fragment重叠!而使用replace只能有一个视图,不像add方式产生多个视图,所以replace在这个场景下不存在视图重叠的情况(从源码角度分析,为什么会发生Fragment重叠?)。
v4-24.0.0+ 开始,官方已经修复了这个问题,使用24.0.0及以上下面两种方案忽略;
网上流行的标准解决方案:
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity);
TargFragment targeFragment ;
HideFragment hideFragment;
if (savedInstanceState != null) { // “内存重启”时调用
targeFragment = getSupportFragmentManager().findFragmentByTag(TargFragment.class.getName);
hideFragment = getSupportFragmentManager().findFragmentByTag(HideFragment.class.getName);
// 解决重叠问题
getFragmentManager().beginTransaction()
.show(targetFragment)
.hide(hideFragment)
.commit();
}else{ // 正常时
targetFragment = TargetFragment.newInstance();
hideFragment = HideFragment.newInstance();
getFragmentManager().beginTransaction()
.add(R.id.container, targetFragment, targetFragment.getClass().getName())
.add(R.id,container,hideFragment,hideFragment.getClass().getName())
.hide(hideFragment)
.commit();
}
}