Fragment的懒加载(转)

1.什么是fragment懒加载以及为什么要使用fragment懒加载。
先看下Demo实现的效果吧。大家对这种效果一定不陌生,知乎,掘金等app都用到了这种效果。
Fragment的懒加载(转)
文章图片
image


这里探索的懒加载是当viewpager 结合tablayout时,多个fragment在进行数据加载的时候,当且仅当fragment对用户可见的时候,才进行数据加载,这里的数据加载可以是网络请求,数据库请求,是需要消耗资源的,如果不使用懒加载,对于那些用户未打开的页面,由于viewpager的加载机制,即使用户未打开的fragment也有可能进行数据加载,造成资源白白的浪费,因此为了良好的用户体验,懒加载是有必要的。
2.fragment懒加载的实现方式以及封装。
fragment懒加载实现的关键在于其的setUserVisibleHint(isVisibleToUser: Boolean)方法,该方法在fragment对用户由可见变为不可见以及由不可见变为可见时都会回调。我们创建抽象AbstractLazyInitFrag,对其进行封装。首先我们引入isVisibleToUser变量,负责保存当前fragment对用户的可见状态。同时还有几个值得注意的地方:
  1. setUserVisibleHint(isVisibleToUser: Boolean)方法的回调时机并没有与fragment的生命周期有确切的关联,比如说,回调时机有可能在onCreateView方法之后,也可能在onCreateView方法之前。因此,必须引入一个标志位isPrepareView判断view是否创建完成,不然,很容易会造成空指针异常。我们初始化该变量为false,在onViewCreated中,也就是view创建完成后,将其赋值为true。代码中是这样:
override fun onViewCreated(view: View, savedInstanceState: Bundle?) { super.onViewCreated(view, savedInstanceState) isPrepareView = true } 复制代码

  1. 数据初始化只应该加载一次,因此,引入第二个标志位,isInitData,初始为false,在数据加载完成之后,将其赋值为true。至此,我们的懒加载方法考虑了所有条件。也就是当isVisibleToUsertrueisInitDatafalseisPrepareViewtrue时,进行数据加载,并且加载后为了防止重复调用,将isInitData赋值为true。代码如下:
private fun lazyInitData() { if (!isInitData && isVisibleToUser && isPrepareView) { isInitData = https://www.it610.com/article/true initData() } } 复制代码

其中initData()为抽象方法,由子类实现,在这里操作数据加载的逻辑。
  1. 该方法的调用时机。首先是 setUserVisibleHint(isVisibleToUser: Boolean)方法中是必须调用的。代码如下:
/*当fragment由可见变为不可见和不可见变为可见时回调*/ override fun setUserVisibleHint(isVisibleToUser: Boolean) { this.isVisibleToUser = isVisibleToUser //标志位保存fragment对用户的可见状态 lazyInitData() } 复制代码

其次,很容易忽略的一点。对于上图中第一个fragment,如果setUserVisibleHint(isVisibleToUser: Boolean)方法在onCreateView之前调用的话,如果懒加载方法只在setUserVisibleHint(isVisibleToUser: Boolean)中调用,那么该fragment将只能在被主动切换一次之后才能加载数据,这肯定是不可能的,因此,我们需要在view创建完成之后,也进行一次调用。思来想去,在onActivityCreated方法中是最合适的。我们在继承的时候,在onViewCreated方法中进行一些初始化就行了,这样不会引起冲突。代码如下:
/*fragment生命周期中onViewCreated之后的方法 在这里调用一次懒加载 避免第一次可见不加载数据*/ override fun onActivityCreated(savedInstanceState: Bundle?) { super.onActivityCreated(savedInstanceState) lazyInitData() }

3.完整代码:
public abstract class BaseLazyFragment extends Fragment { private Context mContext; private View mRootView; private boolean theFragmentVisible; //判断当前fragment是否可见 private boolean isPrepared; //通过此标记位防止空指针 private boolean isInitData=https://www.it610.com/article/false; //此标记位是只能初始化一次数据@Override public void onCreate(@Nullable Bundle savedInstanceState) { super.onCreate(savedInstanceState); mContext=getActivity(); }//该方法会在不同的Fragment切换时才会调用 @Override public void setUserVisibleHint(boolean isVisibleToUser) { if (isVisibleToUser){//isVisibleToUser用于判断当前Fragment是否可见 onVisibleDo(); }else{ onUnVisibleDo(); } super.setUserVisibleHint(isVisibleToUser); }@Nullable @Override public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) { if (mRootView==null){ mRootView=LayoutInflater.from(mContext).inflate(getLayoutId(),null); } return mRootView; }@Override public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) { super.onViewCreated(view, savedInstanceState); //设置布局可见,且数据准备欧克 theFragmentVisible=true; isPrepared=true; //如果setUserVisibleHint()在onCreateView()之前调用。 // 那么该fragment将只能在被主动切换一次之后才能加载数据,这肯定是不可能的,因此,我们需要在view创建完成之后,也进行一次调用 lazyLoad(); }@Override public boolean getUserVisibleHint() { return isVisible(); }private void lazyLoad(){ if (isPrepared && theFragmentVisible && !isInitData){ loadData(); isInitData=true; } } //页面不可见: private void onUnVisibleDo(){ theFragmentVisible=false; }//页面可见: private void onVisibleDo(){ theFragmentVisible=true; lazyLoad(); }//获取子布局id public abstract int getLayoutId(); //加载数据 public abstract void loadData(); }

4.原文出处:
【Fragment的懒加载(转)】原文出处

    推荐阅读