关于Fragment懒加载和ViewPager 使用FragmentStatePagerAdapter跨多个fragment切换空白问题

先来讲一下写这篇文章的原因:就是Fragment懒加载网上的代码弄过来以后配合ViewPager的FragmentPagerAdapter都没问题,但是毕竟使用FragmentPagerAdapter每个fragment都会保持在内存中。而且自己项目中有几十个Fragment所以部分机型就OOM了
接下来附上网上的fragment懒加载代码 在代码中我会注释我改了哪里。

package com.global.motortravel.ui.base;

import android.annotation.TargetApi;
import android.app.Activity;
import android.content.Context;
import android.content.Intent;
import android.os.Build;
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.support.v4.app.Fragment;
import android.util.DisplayMetrics;
import android.util.Log;
import android.view.View;

import com.global.motortravel.R;

public abstract class BaseFragment extends Fragment {
    /**
     * 是否已被加载过一次,第二次就不再去请求数据了
     */
    protected boolean mHasLoadedOnce;
    /**
     * rootView是否初始化标志,防止回调函数在rootView为空的时候触发
     */
    private boolean hasCreateView;

    /**
     * 当前Fragment是否处于可见状态标志,防止因ViewPager的缓存机制而导致回调函数的触发
     */
    private boolean isFragmentVisible;
   //使用属性保存当前状态FragmentStatePagerAdapter 跨好几个页面切换时getUserVisibleHint()在  onViewCreated中获取到的值不准确
    private boolean isVisibleToUser=false;

    protected boolean mHasLoadData=false;//在页面被销毁后是否保持了之前加载的数据,此字段决定是否重新加载数据这样就可以达到FragmentPagerAdapter的效果

    /**
     * onCreateView()里返回的view,修饰为protected,所以子类继承该类时,在onCreateView里必须对该变量进行初始化
     */
    protected View rootView;
    protected int mScreenWidth;
    protected int mScreenHeight;
    public Context context;
    public Activity activity;

    @TargetApi(23)
    @Override
    public void onAttach(Context context) {
        super.onAttach(context);
        onAttachToContext(context);
    }

    /*
     * Deprecated on API 23
     * Use onAttachToContext instead
     */
    @SuppressWarnings("deprecation")
    @Override
    public void onAttach(Activity activity) {
        super.onAttach(activity);
        if (Build.VERSION.SDK_INT < Build.VERSION_CODES.M) {
            onAttachToContext(activity);
        }
        if (this.activity == null) {
            this.activity = activity;
        }
    }

    protected void onAttachToContext(Context context) {
        this.context = context;
    }

    @Override
    public void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        DisplayMetrics dm = getResources().getDisplayMetrics();
        mScreenWidth = dm.widthPixels;
        mScreenHeight = dm.heightPixels;
        if (this.activity == null) {
            this.activity = getActivity();
        }
        initPresenter();
    }


    protected void initEvents() {

    }

    protected void initViews() {

    }

    protected void initPresenter() {

    }


    @Override
    public void onViewCreated(View view, @Nullable Bundle savedInstanceState) {
        super.onViewCreated(view, savedInstanceState);
        if (!hasCreateView && isVisibleToUser) {//网上这里是getUserVisibleHint()这里跨多个fragment切换时取到的值不对所以直接使用刚才在setUserVisibleHint赋值的属性
            onFragmentVisibleChange(true);
            isFragmentVisible = true;
        }
    }

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

    /**
     * 当前fragment可见状态发生变化时会回调该方法
     * 如果当前fragment是第一次加载,等待onCreateView后才会回调该方法,其它情况回调时机跟 {@link #setUserVisibleHint(boolean)}一致
     * 在该回调方法中你可以做一些加载数据操作,甚至是控件的操作,因为配合fragment的view复用机制,你不用担心在对控件操作中会报 null 异常
     *
     * @param isVisible true  不可见 -> 可见
     *                  false 可见  -> 不可见
     */
    protected void onFragmentVisibleChange(boolean isVisible) {

    }

    @Override
    public void setUserVisibleHint(boolean isVisibleToUser) {
        super.setUserVisibleHint(isVisibleToUser);
        this.isVisibleToUser=isVisibleToUser;//跨好几个fragment切换此次的isVisibleToUser正常的所以在这里赋值
        if (rootView == null) {
            return;
        }

        hasCreateView = true;
        if (isVisibleToUser) {
            onFragmentVisibleChange(true);
            isFragmentVisible = true;
            return;
        }
        if (isFragmentVisible) {
            onFragmentVisibleChange(false);
            isFragmentVisible = false;
        }
    }
}

接下来将下我们继承Basefragment的fragment

    @Override
    protected void onFragmentVisibleChange(boolean isVisible) {
      //不可见 或不是第一次加载数据就返回
        if (!isVisible || mHasLoadedOnce) {
            return;
        }
   //页面是否被销毁过 没有销毁过就去加载数据
        if (!mHasLoadData) {
            binding.swipeLayout.setRefreshing(true);
            binding.swipeLayout.postDelayed(new Runnable() {
                @Override
                public void run() {
                    presenter.getHotTravelList(page, pageSize);
                }
            }, 800);
        }
        mHasLoadedOnce = true;
    }

然后就是页面销毁数据保存和页面重新创建直接拿原先的数据

    @Nullable
    @Override
    public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
        //savedInstanceState 不为空说明页面销毁过,把销毁前的数据取回来,这样不要再次加载
        if (savedInstanceState != null) {
            travelInfoList = savedInstanceState.getParcelableArrayList(Constants.TRAVEL_INFO_LIST);
            if (travelInfoList != null && travelInfoList.size() > 0)
              //这里表示数据已经加载无需去请求
                mHasLoadData = true;
        }
        if (rootView == null) {
            binding = DataBindingUtil.inflate(inflater, R.layout.fm_hot_travel, container, false);
            rootView = binding.getRoot();
            initViews();
        }
        return rootView;
    }
    @Override
    public void onSaveInstanceState(Bundle outState) {
        if (adapter.getTravelInfoList() != null) {
          //保存当前fragment的数据
            outState.putParcelableArrayList(Constants.TRAVEL_INFO_LIST, (ArrayList) adapter.getTravelInfoList());
        }
        super.onSaveInstanceState(outState);
    }

大致这么多,同事都是吃饭了我也跟上。上面效果达到 fragment 页面 加载数据 缓存setOffscreenPageLimit(1);你所设置页面 过后销毁 重新创建拉回之前数据也不需要加载,内存还不会随fragment过多而OOM。

你可能感兴趣的:(关于Fragment懒加载和ViewPager 使用FragmentStatePagerAdapter跨多个fragment切换空白问题)