Android TabLayout+ViewPager+Fragment实现懒加载完全解决方案

版权声明:本文为CSDN博主「代码都tm飞了」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接


开发过程中TabLayout配合ViewPager和Fragment的使用是常用的实现多页面的方式。但是这种方式存在一些问题:ViewPager会对其中的Fragment进行预加载。也就是说用户第一次打开第一个界面的时候,不仅第一个界面会进行加载,其他的界面也会进行界面的预加载。这样就会带来界面启动加载慢,浪费系统资源和用户流量的不好的体验。而Fragment的懒加载恰好可以解决这个问题.

首先我们来看看其他App的懒加载实现效果,这里以Bilibili客户端为例:


实现思路:

通过Fragment的setUserVisibleHint方法,这个方法会传递一个boolean类型的参数isVisibleToUser,当Fragment的可见性发生变化时回调这个方法并把是否可见传入参数。所以我们可以在这个方法中进行懒加载。

@Override
public void setUserVisibleHint(boolean isVisibleToUser) {
    super.setUserVisibleHint(isVisibleToUser);
    if (isVisibleToUser) {
        onLazyLoad();//数据加载操作
    }
}

问题1:

setUserVisibleHint方法在切换界面时会多次调用,而我们只希望他被调用一次,既第一次进入页面时被调用。

解决:

在Fragment里创建一个boolean类型的 成员变量isFirstLoad,来判断当前是否为第一次进入界面。并在Fragment初始化的时候将其初始化为true,在setUserVisibleHint方法中进行判断如果为true就开始加载数据并将其置为false,否则不加载。

private boolean isFirstLoad = true;//初始化变量

@Override
    public void setUserVisibleHint(boolean isVisibleToUser) {
        super.setUserVisibleHint(isVisibleToUser);
        if (isFirstLoad && isVisibleToUser) {
            onLazyLoad();//数据加载操作
            isFirstLoad = false;//改变变量的值
        }
    }

问题2:

setUserVisibleHint是Fragment的可见性变化时回调的方法,而onCreateView是Fragment创建视图时回调的方法。但是我们无法保证setUserVisibleHint的调用发生在onCreateView(也就是视图创建)之后。那么我们就是在视图还没有创建时进行数据加载,而往往数据的加载会对视图控件进行操作,那么就会造成空指针的异常发生(因为视图控件还没有初始化)。

解决:

既然我们要保证数据加载发生在视图创建之后,那么我们依然可以通过对isFirstLoad这个成员变量的操作来实现。之前我们是在Fragment创建时就初始化isFirstLoad为true,那么我们这次就可以将他初始化为false然后在onCreateView中将其初始化为true,这样就能保证视图创建完成之前不会进行数据加载的操作。

private boolean isFirstLoad = false;//初始化为false

@Nullable
@Override
public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, Bundle savedInstanceState) {

    View view = inflater.inflate(R.layout.XX,container,false);
    //初始化视图控件...

    isFirstLoad = true;//视图创建完成,将变量置为true

    return view; 
}

    @Override
    public void setUserVisibleHint(boolean isVisibleToUser) {
        super.setUserVisibleHint(isVisibleToUser);
        if (isFirstLoad && isVisibleToUser) {
            onLazyLoad();//数据加载操作
            isFirstLoad = false;
        }
    }

问题3:

setUserVisibleHint这个方法只在Fragment的可见性改变的时候才会被调用,而如果按照上面的代码第一次进入页面之后setUserVisibleHint先被调用,这时视图还没有完成创建,所以数据加载操作不会被调用。而之后没有切换页面,Fragment的可见性也就不会发生改变了,setUserVisibleHint也就不会被调用了,那么数据加载也就不会被执行了。

解决:

既然要保证在视图创建完成后要进行一次数据加载,那么就在onCreateView方法中手动调用一次数据加载就好了。不过这里也要进行对isFirstLoad的操作,在数据加载之前要通过getUserVisibleHint()判断可见性,在加载后还要将isFirstLoad置为false。这样才足够严谨。

private boolean isFirstLoad = false;

@Nullable
@Override
public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, Bundle savedInstanceState) {

    View view = inflater.inflate(R.layout.XX,container,false);
    //初始化视图控件...

    isFirstLoad = true;//视图创建完成,将变量置为true 

    if (getUserVisibleHint()) {//判断Fragment是否可见
        onLazyLoad();//数据加载操作
        isFirstLoad = false;//将变量置为false
    }
    return view;
}

@Override
    public void setUserVisibleHint(boolean isVisibleToUser) {
        super.setUserVisibleHint(isVisibleToUser);
        if (isFirstLoad && isVisibleToUser) {
            onLazyLoad();//数据加载操作
            isFirstLoad = false;//将变量置为false
        }

    }

完全解决方案:

创建一个抽象父类:BaseLazyLoadFragment,在父类中实现整个懒加载的流程,然后将初始化视图、初始化事件和数据加载的接口暴露给子类让子类来实现。完整代码:

public abstract class BaseLazyLoadFragment extends Fragment {

    private boolean isFirstLoad = false;

    @Nullable
    @Override
    public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, Bundle savedInstanceState) {

        View view = initView(inflater, container);//让子类实现初始化视图

        initEvent();//初始化事件

        isFirstLoad = true;//视图创建完成,将变量置为true

        if (getUserVisibleHint()) {//如果Fragment可见进行数据加载
            onLazyLoad();
            isFirstLoad = false;
        }
        return view;
    }

    @Override
    public void onDestroyView() {
        super.onDestroyView();
        isFirstLoad = false;//视图销毁将变量置为false
    }

    @Override
    public void setUserVisibleHint(boolean isVisibleToUser) {
        super.setUserVisibleHint(isVisibleToUser);
        if (isFirstLoad && isVisibleToUser) {//视图变为可见并且是第一次加载
            onLazyLoad();
            isFirstLoad = false;
        }

    }
    //数据加载接口,留给子类实现
    public abstract void onLazyLoad();
    
    //初始化视图接口,子类必须实现
    public abstract View initView(LayoutInflater inflater, @Nullable ViewGroup container);
    
    //初始化事件接口,留给子类实现
    public abstract void initEvent();

}

使用:

如果当前的Fragment要用到懒加载功能的话,只要让他继承自BaseLazyLoadFragment,并重写对应的方法就可以了。

效果展示:

这里只有一个Activity,然后通过TabLayout+ViewPager+Fragment来实现多页面,接着用BaseLazyLoadFragment来实现懒加载。这里用SwipeRefreshLayout的加载和视图的隐藏/显示来模拟数据加载过程。

注意:

ViewPager默认只保留当前视图的前后各一个视图,其他的视图会被销毁。如果不想让视图被销毁要重写FragmentPagerAdapter的destroyItem方法,并注释掉原本的代码。

总结:

其实之所以Fragment的懒加载这么麻烦是因为Fragment的生命周期很特殊,Fragment的几个生命周期方法都是和他所绑定的Activity的生命周期对应的。但是当Activity可见的时候Fragment却不一定可见,所以出现了Fragment专有的生命周期方法:setUserVisibleHint()。但是在使用时也不可以脱离Activity的那几个生命周期来使用,所以懒加载实现起来才有这么多讲究。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,634评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,951评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,427评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,770评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,835评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,799评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,768评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,544评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,979评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,271评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,427评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,121评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,756评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,375评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,579评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,410评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,315评论 2 352