前言
前段时间在工作中遇到了fragment的重叠问题,耽误了挺久的,主要原因还是在于自己对fragment的研究还不够深入,只是一些泛泛的了解。
重叠问题出现的原因
当我使用show和hide来控制fragment的显隐时,一旦作为容器的activity被系统GC,导致activity的所有生命周期重新被调用,两个fragment便会重叠出错。
原因是当activity被意外回收时会调用onSaveInstanceState()保持自身的一些状态和信息,以备在onCreate()方法中能够直接获取而不需要重新创建。
fragment的实例也会在onSaveInstanceState()中进行保存,所以在onCreate()方法中等于创建了两个Fragment的实例,当我控制一个fragment隐藏时,假设这个fragment的名字叫aFragment,由于现在activity实际上有两个aFragment,而我只隐藏了一个所以会导致fragment重叠。
解决的办法
既然原因是由于创建了两次Fragment导致的,那么直觉的办法当然是控制activity的onCreate()方法让其只创建一次Fragment。
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
// 在页面重启时,Fragment会被保存恢复,而此时再加载Fragment会重复加载,导致重叠 ;
if(saveInstanceState == null){
// 或者 if(findFragmentByTag(mFragmentTag) == null)
// 正常情况下去 加载根Fragment
}
}
通过判断savedInstanceState是否为空可以知道这是第一次创建还是GC之后再次调用的onCreate(),只在第一次onCreate()时创建aFragmen的实例,就可以避免重叠的问题。
还有一个办法就是重写activity的onSaveInstanceState()方法,去掉其中保存fragment的代码,缺点是当页面比较复杂时,saveInstance除了保存fragment的状态还保存了所有view的状态,必须将view的状态保存写一遍,所以推荐使用第一种方法。