Android ViewPager + Fragment 懒加载实现

3,878 阅读2分钟


ViewPager + Fragment 实现左右切换的页面,默认会加载当前Fragment和相邻的Fragment数据,为了更好的用户体验,最好是切到当前Fragment再加载数据,这样会为用户节约很多流量,另外ViewPager的setOffscreenPageLimit(n)可以设置缓存页面,当前页面的相邻n各页面都会被缓存。做到当前Fragment加载数据需要重写setUserVisibleHint方法,一般网上Fragment 懒加载实现方法都是如下这样的:


protected boolean isVisible;
@Override
public void setUserVisibleHint(boolean isVisibleToUser) {
    super.setUserVisibleHint(isVisibleToUser);
    if (getUserVisibleHint()) {
        isVisible = true;
        onVisible();
    } else {
        isVisible = false;
        onInvisible();
    }
}
protected void onVisible() {
    lazyLoad();
}
protected void lazyLoad() {
    if (!isVisible) {
        return;
    }
    //getData();//数据请求
}
protected void onInvisible() {
}

对,这样确实实现了懒加载,然后实际开发中,请求数据完毕,会绑定RecyclerView数据源,结果会发现报空指针异常,原来是setUserVisibleHint方法会比onCreateView先走,RecyclerView肯定是null,基于此,我们可以再添加一个标志位,判断初始化是否完成,然后再去请求数据,这样就不会有问题了,完整代码如下:


public class WeiYanFragment extends Fragment {
    // 标志位,标志已经初始化完成,因为setUserVisibleHint是在onCreateView之前调用的,
    // 在视图未初始化的时候,在lazyLoad当中就使用的话,就会有空指针的异常
    private boolean isPrepared;
    //标志当前页面是否可见
    private boolean isVisible;
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,Bundle savedInstanceState) {
        return inflater.inflate(R.layout.weiyan, container, false);
    }
    @Override
    public void onViewCreated(View view, @Nullable Bundle savedInstanceState) {
        super.onViewCreated(view, savedInstanceState);
        isPrepared = true;
        lazyLoad();
    }
    @Override
    public void setUserVisibleHint(boolean isVisibleToUser) {
        super.setUserVisibleHint(isVisibleToUser);
        //懒加载
        if (getUserVisibleHint()) {
            isVisible = true;
            onVisible();
        } else {
            isVisible = false;
            onInvisible();
        }
    }
    protected void onVisible() {
        lazyLoad();
    }
    protected void lazyLoad() {
        if (!isVisible || !isPrepared) {
            return;
        }
        //getData();//数据请求
    }
    protected void onInvisible() {
    }
}