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。

你可能感兴趣的:(Fragment使用过程中遇到过的坑)