Viewpager+Fragment潜在的一些坑及解决办法

viewpager加fragment的搭配在平常的开发中很常见,这样可以很好的避免的fragment单独使用时在切换页面时常常遇到的一些显示问题又可以快速的解决界面数据预加载保障流畅度。然而世间万物都有两面性,viewpager+fragment虽然有时大大的,但是如果使用的不正确,也会带来一些莫名其妙的问题,接下来我们就对这俩控件的使用最一个全面的整理。

固定子页数量预加载方式:

//创建数据
List fragmentList = new ArrayList<>();
FragmentA fragmentA = new FragmentA();
fragmentList.add(fragmentA);
FragmentB fragmentB = new FragmentB();
fragmentList.add(fragmentB);

//adapter
public class MyPagerAdapter extends FragmentPagerAdapter {
    private List fragmentList = null;
    public void setList(List fragmentList) {
        this.fragmentList = fragmentList;
    }
    public MyPagerAdapter(FragmentManager fm) {
        super(fm);
    }
    @Override
    public Fragment getItem(int position) {
        return fragmentList.get(position);
    }
    @Override
    public int getCount() {
        return fragmentList.size();
    }
}
//设置数据
Viewpager viewpager = findViewById(R.id.viewpager);
//设置缓存容量
viewpager.setOffscreenPageLimit(fragmentList.size());
//这里的FragmentManager取值要小心了,如果在activity中设置PagerAdapter的话,使用getSupportFragmentManager(),如果在fragment中设置的话,则使用getChildFragmentManager(),否则子页将不显示
//MyPagerAdapter adapter = new MyPagerAdapter(getSupportFragmentManager());
MyPagerAdapter adapter = new MyPagerAdapter(getChildFragmentManager());
viewpager.setAdapter(adapter);

默认预加载方式(默认缓存3页):

/**
 * 这里必须要用FragmentStatePagerAdapter,不然用 notifyDataSetChanged方法不会刷新数据
 */
public class MyFgAdapter extends FragmentStatePagerAdapter {
  int count = 0;
  public MyFgAdapter(FragmentManager fm) {
        super(fm);
  }
  public void setCount(int count) {
        this.count = count;
  }
  //重写此方法才能支持刷新数据
  @Override
  public void notifyDataSetChanged() {
        mChildCount = getCount();
        super.notifyDataSetChanged();
  }
  int mChildCount = 0;
  @Override
  public int getItemPosition(Object object) {
        if (mChildCount > 0) {
            mChildCount--;
            return POSITION_NONE;
        }
        return super.getItemPosition(object);
  }
  @Override
  public ChildFragment getItem(int position) {
  //这里在viewpager滑动时会随着position的变化而决定是否销毁,在销毁一个子页后,在滑动时控件会默认自动走到这个方法内创建一个子页,这时候我们就可以在这个方法内进行自定义操作
        ChildFragment childFragment = new ChildFragment();
        return childFragment;
    }
    @Override
    public int getCount() {
        return count;
    }
 }
//设置数据
Viewpager viewpager = findViewById(R.id.viewpager);
//这里的FragmentManager取值要小心了,如果在activity中设置PagerAdapter的话,使用getSupportFragmentManager(),如果在fragment中设置的话,则使用getChildFragmentManager(),否则子页将不显示
//MyFgAdapter adapter = new MyFgAdapter(getSupportFragmentManager());
if(adapter==null){
    adapter = new MyFgAdapter(getChildFragmentManager());
    viewpager.setAdapter(adapter);
}else{
    //这里通过调用FragmentStatePagerAdapter的notifyDataSetChanged实现子页的全部刷新
    viewpager.setCurrentItem(0);
    adapter.setCount(10);
    adapter.notifyDataSetChanged();
}

小结:
通过上面的代码和注释说明展示,可以看出以下几点:
1.FragmentPagerAdapter不支持刷新,FragmentStatePagerAdapter支持刷新;

2.getSupportFragmentManager()在activity中使用,getChildFragmentManager()在fragment中使用;

3.固定viewpager子页数量时可以使用setOffscreenPageLimit()设置缓存数量,使用默认缓存时,不能使用setOffscreenPageLimit();

你可能感兴趣的:(界面布局,控件)