Saturday, September 10, 2016

fragment的懒加载实现

我们在做应用开发的时候,一个Activity里面可能会以viewpager(或其他容器)与多个Fragment来组合使用,而如果每个fragment都需要去加载数据,或从本地加载,或从网络加载,那么在这个activity刚创建的时候就变成需要初始化大量资源。这样的结果,我们当然不会满意。那么,能不能做到当切换到这个fragment的时候,它才去初始化呢?
答案就在Fragment里的setUserVisibleHint这个方法里。请看关于Fragment里这个方法的API文档(国内镜像地址:Fragment api):
  1. Set a hint to the system about whether this fragment's UI is currently visible to the user. This hint defaults to true and is persistent across fragment instance state save and restore.
  2. An app may set this to false to indicate that the fragment's UI is scrolled out of visibility or is otherwise not directly visible to the user. This may be used by the system to prioritize operations such as fragment lifecycle updates or loader ordering behavior.
  3. Parameters
  4. isVisibleToUser true if this fragment's UI is currently visible to the user (default), false if it is not.

该方法用于告诉系统,这个Fragment的UI是否是可见的。所以我们只需要继承Fragment并重写该方法,即可实现在fragment可见时才进行数据加载操作,即Fragment的懒加载。
代码如下:

  1. /*
  2.  * Date: 14-7-17
  3.  * Project: Access-Control-V2
  4.  */
  5. package cn.irains.access_control_v2.common;
  6. import android.support.v4.app.Fragment;
  7. /**
  8.  * Author: msdx (645079761@qq.com)
  9.  * Time: 14-7-17 下午5:46
  10.  */
  11. public abstract class LazyFragment extends Fragment {
  12.     protected boolean isVisible;
  13.     /**
  14.      * 在这里实现Fragment数据的缓加载.
  15.      * @param isVisibleToUser
  16.      */
  17.     @Override
  18.     public void setUserVisibleHint(boolean isVisibleToUser) {
  19.         super.setUserVisibleHint(isVisibleToUser);
  20.         if(getUserVisibleHint()) {
  21.             isVisible = true;
  22.             onVisible();
  23.         } else {
  24.             isVisible = false;
  25.             onInvisible();
  26.         }
  27.     }
  28.     protected void onVisible(){
  29.         lazyLoad();
  30.     }
  31.     protected abstract void lazyLoad();
  32.     protected void onInvisible(){}
  33. }
在LazyFragment,我增加了三个方法,一个是onVisiable,即fragment被设置为可见时调用,一个是onInvisible,即fragment被设置为不可见时调用。另外再写了一个lazyLoad的抽象方法,该方法在onVisible里面调用。你可能会想,为什么不在getUserVisibleHint里面就直接调用呢?
我这么写是为了代码的复用。因为在fragment中,我们还需要创建视图(onCreateView()方法),可能还需要在它不可见时就进行其他小量的初始化操作(比如初始化需要通过AIDL调用的远程服务)等。而setUserVisibleHint是在onCreateView之前调用的,那么在视图未初始化的时候,在lazyLoad当中就使用的话,就会有空指针的异常。而把lazyLoad抽离成一个方法,那么它的子类就可以这样做:
  1. public class OpenResultFragment extends LazyFragment{
  2.     // 标志位,标志已经初始化完成。
  3.     private boolean isPrepared;
  4.     @Override
  5.     public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
  6.         Log.d(LOG_TAG, "onCreateView");
  7.         View view = inflater.inflate(R.layout.fragment_open_result, container, false);
  8.         //XXX初始化view的各控件
  9.     isPrepared = true;
  10.         lazyLoad();
  11.         return view;
  12.     }
  13.     @Override
  14.     protected void lazyLoad() {
  15.         if(!isPrepared || !isVisible) {
  16.             return;
  17.         }
  18.         //填充各控件的数据
  19.     }
  20. }
在上面的类当中,我们增加了一个标志位isPrepared,用于标志是否初始化完成。然后在我们所需要的初始化操作完成之后调用,如上面的例子当中,在初始化view之后,设置 isPrepared为true,同时调用lazyLoad()方法。而在lazyLoad()当中,判断isPrepared和isVisible只要有一个不为true就不往下执行。也就是仅当初始化完成,并且可见的时候才继续加载,这样的避免了未初始化完成就使用而带来的问题。
在这里我对fragment的懒加载实现的介绍就到此为止,如果你有兴趣,可以基于此再深入探究,比如写一个带有缓初始化和可见时刷新的特性的Fragment。

No comments:

Post a Comment