Fragment使用过程中遇到过的坑
使用Fragment可以让APP更加的流畅,并且降低内存占用,同样的界面Activity占用内存比Fragment要多,相应速度Fragment比Activity在中低端手机上快了很多。但是在Fragment的使用过程中总会出现一些bug,比如Fragment嵌套或者是单Activity+多Fragment架构时。
1.getActivity()空指针
可能你会遇到过在Fragment中使用getActivity时返回null,大多数情况下的原因是:你在调用了getActivity时,当前的Fragment已经onDetach()了宿主的Activity。
解决方法:在Fragment基类里设置一个Activity mActivity的全局变量,在onAttach里面复制,使用mActivity代替getActivity,保证Fragment即使在onDetach后,仍持有Activity引用。
protected Activity mActivity;
@Override
public void onAttach(Activity activity) {
super.onAttach(activity);
this.mActivity = activity;
}
/**
- 如果你用了support 23的库,上面的方法会提示过时,有强迫症的小伙伴,可以用下面的方法代替
*/
@Override
public void onAttach(Context context) {
super.onAttach(context);
this.mActivity = (Activity)context;
}
2.Fragment重叠异常
如果你add()了几个Fragment,使用show()、hide()方法控制,比如微信、qq底部tab等场景,如果你什么都不做的话,在“内存重启”后回到前台,app的这几个Fragment界面会重叠。
原因是FragmentManager帮我们管理Fragment,每当我们离开该Activity,FragmentManager都会保存它的Fragments,当发生“内存重启”,他会从栈低向栈顶回复Fragment,并且全部都是以show的方式,所以我们看到了界面重叠。
解决方法:
(1)即在add()或者replace()时绑定一个tag,一般我们使用相应Fragment的类名作为tag,然后在发生“内存重启”时,通过findFragmentByTag找到相应的Fragment,并hide()需要隐藏的Fragment。
标准写法:
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity);
TargetFragment targetFragment;
HideFragment hideFragment;
if (savedInstanceState != null) { // “内存重启”时调用
targetFragment = getSupportFragmentManager().findFragmentByTag(targetFragment.getClass().getName);
hideFragment = getSupportFragmentManager().findFragmentByTag(hideFragment.getClass().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();
}
}
如果你想恢复到用户离开时的那个Fragment界面,你还需要在onSaveInstanceState里保存离开时的那个Fragment的tag或者下标,在onCreate“内存重启”代码中取出tag或者下标,进行恢复。
(2)使用getSupportFragmentManager.getFragments()恢复。
通过getFragments()可以获取当前FragmentManager管理的栈内所有Fragment
标准写法如下:
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity);
TargetFragment targetFragment;
HideFragment hideFragment;
if (savedInstanceState != null) { // “内存重启”时调用
List fragmentList = getSupportFragmentManager().getFragments();
for (Fragment fragment : fragmentList) {
if(fragment instanceof TartgetFragment){
targetFragment = (TargetFragment)fragment;
}else if(fragment instanceof HideFragment){
hideFragment = (HideFragment)fragment;
}
}
// 解决重叠问题
getFragmentManager().beginTransaction()
.show(targetFragment)
.hide(hideFragment)
.commit();
}else{ // 正常时
targetFragment = TargetFragment.newInstance();
hideFragment = HideFragment.newInstance();
// 这里add时,tag可传可不传
getFragmentManager().beginTransaction()
.add(R.id.container)
.add(R.id,container,hideFragment)
.hide(hideFragment)
.commit();
}
}
从代码看起来这种方法比较复杂,但是这种方法在一些场景下比第一种方法更加简单有效。顺便提一下,有些小伙伴会用一种并不合适的方法恢复Fragment,虽然效果也能达到,但并不恰当。
// 保存
@Override
protected void onSaveInstanceState(Bundle outState) {
super.onSaveInstanceState(outState);
getSupportFragmentManager().putFragment(outState, KEY, targetFragment);
}
// 恢复
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_scrolling);
if (savedInstanceState != null) {
Fragment targetFragment = getSupportFragmentManager().getFragment(savedInstanceState, KEY);
}
}
如果仅仅为了找回栈内的Fragment,使用putFragment保存Fragment是完全没有必要的。因为FragmentManager在任何情况都会把你存储Fragment,你要做的仅仅是在“内存重启”后找到这些Fragment。