【Android】Fragment懒加载和ViewPager的坑

本篇文章已受权微信公众号 dasu_Android(大苏)独家发布html

效果

老规矩,先来看看效果java

ANDROID和福利两个Fragment是设置的Fragment可见时加载数据,也就是懒加载。圆形的旋转加载图标只有一个,因此,若是当前Fragment正处于加载状态,在离开该Fragment时须要隐藏加载动画,由于另外一个Fragment并不必定处于加载状态,当返回Fragment时,若是仍是处于加载状态,则要能够实现自动显示加载动画,若是数据已经加载完毕则不须要再显示出来。android

以上效果就是今天要介绍和分享的,那么开始往下看吧。git

懒加载

懒加载意思也就是当须要的时候才会去加载github

那么,为何Fragment须要懒加载呢,通常咱们都会在onCreate()或者onCreateView()里去启动一些数据加载操做,好比从本地加载或者从服务器加载。大部分状况下,这样并不会出现什么问题,可是当你使用ViewPager + Fragment的时候,问题就来了,这时就应该考虑是否须要实现懒加载了。api

ViewPager + Fragment 的坑

ViewPager为了让滑动的时候能够有很好的用户的体验,也就是防止出现卡顿现象,所以它有一个缓存机制。默认状况下,ViewPager会提早建立好当前Fragment旁的两个Fragment,举个例子说也就是若是你当前显示的是编号3的Fragment,那么其实编号2和4的Fragment也已经建立好了,也就是说这3个Fragment都已经执行完 onAttach() -> onResume() 这之间的生命周期函数了。缓存


原本Fragment的 onResume()表示的是当前Fragment处于可见且可交互状态,但因为ViewPager的缓存机制,它已经失去了意义,也就是说咱们只是打开了“福利”这个Fragment,但其实“休息视频”和“拓展资源”这两个Fragment的数据也都已经加载好了。服务器

若是加载数据的操做都比较耗时或者都是相似图片的占用大量内存,这时就应该考虑想一想是否该实现懒加载。也就是,当我打开哪一个Fragment的时候,它才会去加载数据。微信

懒加载实现?

setUserVisibleHint(boolean isVisibleToUser) 能够?

当你去网上查找相关资料时,你会发现不少人推荐说把加载数据的操做放在这个函数里,isVisibleToUser表示当前Fragment是否可见。那么,是否真的能够就这样作呢?先来看个日志:架构

题主是从 DayDataFragment 跳转到 MeiziDataFragment 的,因此能够看到日志里面:DayDataFragment打出了false,表示它不可见了。而MeiziDataFragment却先打出了false,而后才打出true,这是由于setUserVisibleHint()在Fragment实例化时会先调用一次,而且默认值是false,当选中当前显示的Fragment时还会再调用一次。

因此,看上面的日志,除了DayDataFragment外,其余三个Fragment均没有实例化,因此当打开MeiziDataFragment时,由于ViewPager的缓存机制,会同时建立三个Fragment的实例,因此打印了三条isVisibleToUeser: false的日志,由于选中的是MeiziDataFragment,因此它还会触发一次setUserVisibleHint(),而且打印出true。

那么,是否能够在setUserVisibleHint(boolean isVisibleToUser)里进行数据加载操做来实现懒加载呢?

能够是能够,若是你只是须要数据的懒加载的话,但若是你还有如下的需求,那么这种方式就不行了:

一、若是你在Fragment可见时须要进行一些控件的操做,好比显示加载控件

二、若是你还须要在Fragment从 “可见 -> 不可见” 时进行一些操做的话,好比取消加载控件显示

这边再提一下,setUserVisibleHint()可能会在Fragment的生命周期以外被调用,也就是可能在view建立前就被调用,也可能在destroyView后被调用,因此若是涉及到一些控件的操做的话,可能会报 null 异常,由于控件还没初始化,或者已经摧毁了。

进一步封装

题主稍微进行了一些封装,自定义了一个新的回调函数onFragmentVisibleChange(boolean isVisible),能够实现的效果有:

一、只有两种状况会触发该函数

二、一种是Fragment从“不可见 -> 可见” 时触发,并传入 isVisible = true

三、一种是Fragment从“可见 -> 不可见” 时触发,并传入 isVisible = false

四、能够在该函数内进行控件的操做,不会报null异常

题主此次仍旧是从DayDataFragment 跳转到 MeiziDataFragment, 但跟上上面的日志图片不一样,这里只打印了两条日志,也就是说即便有三个Fragment被实例化了,但只有显示的那个Fragment和离开的那个Fragment才会触发回调函数,这样就能够支持咱们在可见状态变化时进行一些操做,由于不会有多余的false触发。

另外,由于ViewPager缓存机制,因此题主进行了view的复用,防止onCreateView()重复的建立view,其实也就是将view设置为成员变量,建立view时先判断是否为null。由于ViewPager里对Fragment的回收和建立时,若是Fragment已经建立过了,那么只会调用 onCreateView() -> onDestroyView() 生命函数,onCreate()和onDestroy并不会触发,因此关于变量的初始化和赋值操做能够在onCreate()里进行,这样就能够避免重复的操做。

代码


2016-04-21 更新:该博客封装的懒加载实现有些不足,好比不支持数据只有第一次打开Fragment时才进行加载的应用场景,所以从新写了篇博客,能够移步至此观看:再来一篇Fragment的懒加载(只加载一次哦)


最后附上代码,另外注意一下,题主是从项目里抽出代码,进行一些修改,让它尽可能能够直接复制粘贴使用,但并无进行过测试,因此若是不行的话能够留言,题主会查看。或者你直接到我原项目里去查看,代码已托管至Github上,由于项目是针对具体需求的,因此类里面会增长不少其余无关的代码。再或者,你能够尝试本身进行封装下,代码不多,不到50行,理解思路就好了。

/**
 * Created by dasu on 2016/9/27.
 * https://github.com/woshidasusu/Meizi
 *
 * Viewpager + Fragment状况下,fragment的生命周期因Viewpager的缓存机制而失去了具体意义
 * 该抽象类自定义一个新的回调方法,当fragment可见状态改变时会触发的回调方法,介绍看下面
 *
 * @see #onFragmentVisibleChange(boolean)
 */
public abstract class ViewPagerFragment extends Fragment {

    /**
     * rootView是否初始化标志,防止回调函数在rootView为空的时候触发
     */
    private boolean hasCreateView;
    
    /**
     * 当前Fragment是否处于可见状态标志,防止因ViewPager的缓存机制而致使回调函数的触发
     */
    private boolean isFragmentVisible;
    
    /**
     * onCreateView()里返回的view,修饰为protected,因此子类继承该类时,在onCreateView里必须对该变量进行初始化
     */
    protected View rootView;

    @Override
    public void setUserVisibleHint(boolean isVisibleToUser) {
        super.setUserVisibleHint(isVisibleToUser);
        Log.d(getTAG(), "setUserVisibleHint() -> isVisibleToUser: " + isVisibleToUser);
        if (rootView == null) {
            return;
        }
        hasCreateView = true;
        if (isVisibleToUser) {
            onFragmentVisibleChange(true);
            isFragmentVisible = true;
            return;
        }
        if (isFragmentVisible) {
            onFragmentVisibleChange(false);
            isFragmentVisible = false;
        }
    }

    @Override
    public void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        initVariable();
    }

    @Override
    public void onViewCreated(View view, @Nullable Bundle savedInstanceState) {
        super.onViewCreated(view, savedInstanceState);
        if (!hasCreateView && getUserVisibleHint()) {
            onFragmentVisibleChange(true);
            isFragmentVisible = true;
        }
    }

    private void initVariable() {
        hasCreateView = false;
        isFragmentVisible = false;
    }

    /**************************************************************
     *  自定义的回调方法,子类可根据需求重写
     *************************************************************/

    /**
     * 当前fragment可见状态发生变化时会回调该方法
     * 若是当前fragment是第一次加载,等待onCreateView后才会回调该方法,其它状况回调时机跟 {@link #setUserVisibleHint(boolean)}一致
     * 在该回调方法中你能够作一些加载数据操做,甚至是控件的操做,由于配合fragment的view复用机制,你不用担忧在对控件操做中会报 null 异常
     *
     * @param isVisible true  不可见 -> 可见
     *                  false 可见  -> 不可见
     */
    protected void onFragmentVisibleChange(boolean isVisible) {
        Log.w(getTAG(), "onFragmentVisibleChange -> isVisible: " + isVisible);
    }
}

用法

新建类ViewPagerFragment,将上面代码复制粘贴进去,添加须要的import语句 -> 新建你须要的Fragment类,继承ViewPagerFragment,在onCreateView()里对rootView进行初始化 -> 重写onFragmentVisibleChange(),在这里进行你须要的操做,好比数据加载,控制显示等。

@Nullable
    @Override
    public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
        if (rootView == null) {
            rootView = inflater.inflate(R.layout.fragment_android, container, false);
        
        }
        return rootView;
    } 
    
    @Override
    protected void onFragmentVisibleChange(boolean isVisible) {
        super.onFragmentVisibleChange(isVisible);
        if (isVisible) {
        //   do things when fragment is visible    
        //    if (ListUtils.isEmpty(mDataList) && !isRefreshing()) {
        //        setRefresh(true);
        //        loadServiceData(false);
            } else {
        //        setRefresh(false);
            }
        }
    }

项目Github 地址

PS

以上就是此次的内容了,最近题主想利用 Gank公开的api,作一个相似于Meizi的应用出来,虽然这个App已经有无数人都作过了,但确实是一个很值得学习的项目,题主仍然是小白一个,因此仍是好好学习下。drakeet的Meizi项目用到了不少高级技术,好比Rxjava之类的,题主看不懂,其余Github上一些比较出名的Meizi App要么是MVP架构,要么仍是用到了目前小白的我看懂的技术,因此此次就决定本身用最基础的MVC,一些简单经常使用的第三方库来作这个App,毕竟路要一步一步走,若是这个完成了,收获和体验应该会不少(这不就收获了这篇随笔了吗O(∩_∩)O),因此,若是有兴趣的话,欢迎Start,欢迎指点,欢迎拍砖,你们一块儿学习进步。


QQ图片20180316094923.jpg 最近刚开通了公众号,想激励本身坚持写做下去,初期主要分享原创的Android或Android-Tv方面的小知识,感兴趣的能够点一波关注,谢谢支持~~

相关文章
相关标签/搜索