fragment懒加载

作者: 一杯刘 | 来源:发表于2017-04-17 22:14 被阅读198次

    网上关于懒加载的文章很多很多,这里总结一下,并给出个人的改装方案

    什么是懒加载?

    ViewPager + Fragment 的场景下,由于 ViewPager 默认(也就是最少)加载 3 个页面,那如果 Fragment 初始化需要做大量的耗时操作,势必会造成卡顿,于是乎,人们提出来懒加载这个概念,即在页面显示的时候才加载相应 Fragment,也就降低了卡顿的可能性。

    实现方法

    利用 setUserVisibleHint(boolean isVisibleToUser) 方法和 getUserVisibleHint 方法

    关于两者的说明以及我接下来的封装与改造,都基于这里

    我的思路是把懒加载的代码放在 BaseFragment 中,这样你的 Fragment 只要继承他就好了,在实现类中,重写 initView() 和 load() 方法即可,initView 方法就相当于 onCreateView,在这里进行控件初始化工作,而 load 方法就是你需要加载数据的位置啦。

    解释都在注释中

    talk is cheap show you the code

    public abstract class BaseFragment extends Fragment {
    
        protected boolean mIsUserVisible;    // fragment 是否可见
        protected boolean mIsFragmentCreateView = false;  // fragment 是否创建完毕
        /**
        * 是否需要重新加载
        * 比如页面1、2、3,刚开始在 1 页面,滑动到 2 页面,
        * 此时 1 页面还在 ViewPager 的缓存中,
        * 那么当我又回到 1 页面,此时不应该重新加载
        */
        protected boolean mIsNeedReLoaded = true;
    
        @Override
        public void setUserVisibleHint(boolean isVisibleToUser) {
            super.setUserVisibleHint(isVisibleToUser);
            // getUserVisibleHint() 返回该 fragment 是否可见
            if (getUserVisibleHint()) {
                mIsUserVisible = true;   // 可见
                onVisible();
            } else {
                mIsUserVisible = false;  // 不可见
                onInvisible();
            }
        }
    
        protected void onVisible() {
          // 执行懒加载的代码
            lazyLoad();
        }
    
        protected void lazyLoad() {
          /**
          *  此三个条件 保证
          *  1.fragment 执行到 onCreateView,即 控件初始化完毕
          * (由于 setUserVisibleHint 的调用时机不确定,防止控件未初始化就进行
          *                   操作,防止空指针异常)
          *  2.该 Fragment 是否可见
          *  3.是否需要加载(缓存中的页面不需要重新加载)
          */
            if (!mIsFragmentCreateView || !mIsUserVisible || mIsNeedReLoaded) {
                return;
            }
            // 已经加载过一次,在该 fragment 未销毁的时候,不应该再次加载
            mIsNeedReLoaded = false;  
            load();
        }
    
        protected void onInvisible() {
    
        }
    
        @Override
        public View onCreateView(LayoutInflater inflater,
                                ViewGroup container, Bundle savedInstanceState) {
    
            // 加载控件
            View view = initView(inflater, container,savedInstanceState);
    
            mIsFragmentCreateView = true;  // 控件已经加载好了
    
            lazyLoad();
    
            return view;
        }
    
        // 在这里初始化控件
        protected abstract View initView(LayoutInflater inflater,
                                ViewGroup container,Bundle savedInstanceState);
        // 这里就是懒加载的位置啦
        protected abstract void load();
    
        @Override
        public void onDestroyView() {
            super.onDestroyView();
            //
            mIsFragmentCreateView = false;  // 控件都销毁了
            mIsNeedReLoaded = true;  // 页面都销毁了,需要重新加载
        }
    }
    

    注释很清楚了吧,那这次就到这里吧,懒加载本身也不是很难的东西,多看看就好了(有问题欢迎留言)。

    下一篇准备写一写 Retrofit + Rxjava2.0 + MVP 三基佬不得不说的故事...

    またにしましょう

    相关文章

      网友评论

        本文标题:fragment懒加载

        本文链接:https://www.haomeiwen.com/subject/wzsxzttx.html