Android Viewpager+Fragment的懒加载优化问题

我们在做应用开发的时候,一个Activity里面可能会以viewpager(或其他容器)与多个Fragment来组合使用,而如果每个fragment都需要去加载数据,或从本地加载,或从网络加载,那么在这个activity刚创建的时候就变成需要初始化大量资源。这样的结果,我们当然不会满意。那么,能不能做到当切换到这个fragment的时候,它才去初始化呢?

答案就在Fragment里的setUserVisibleHint这个方法里。请看关于Fragment里这个方法的API文档(国内镜像地址:Fragment api):

1
2
3
4
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.
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.
Parameters
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
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
/*
  * Date: 14-7-17
  * Project: Access-Control-V2
  */
package cn.irains.access_control_v2.common;
import android.support.v4.app.Fragment;
/**
  * Author: msdx (645079761@qq.com)
  * Time: 14-7-17 下午5:46
  */
public abstract class LazyFragment extends Fragment {
     protected boolean isVisible;
     /**
      * 在这里实现Fragment数据的缓加载.
      * @param isVisibleToUser
      */
     @Override
     public void setUserVisibleHint(boolean isVisibleToUser) {
         super .setUserVisibleHint(isVisibleToUser);
         if (getUserVisibleHint()) {
             isVisible =  true ;
             onVisible();
         else  {
             isVisible =  false ;
             onInvisible();
         }
     }
     protected void onVisible(){
         lazyLoad();
     }
     protected abstract void lazyLoad();
     protected void onInvisible(){}
}

在LazyFragment,我增加了三个方法,一个是onVisiable,即fragment被设置为可见时调用,一个是onInvisible,即fragment被设置为不可见时调用。另外再写了一个lazyLoad的抽象方法,该方法在onVisible里面调用。你可能会想,为什么不在getUserVisibleHint里面就直接调用呢?

我这么写是为了代码的复用。因为在fragment中,我们还需要创建视图(onCreateView()方法),可能还需要在它不可见时就进行其他小量的初始化操作(比如初始化需要通过AIDL调用的远程服务)等。而setUserVisibleHint是在onCreateView之前调用的,那么在视图未初始化的时候,在lazyLoad当中就使用的话,就会有空指针的异常。而把lazyLoad抽离成一个方法,那么它的子类就可以这样做:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
public class OpenResultFragment extends LazyFragment{
     // 标志位,标志已经初始化完成。
     private boolean isPrepared;
     @Override
     public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
         Log.d(LOG_TAG,  "onCreateView" );
         View view = inflater.inflate(R.layout.fragment_open_result, container,  false );
         //XXX初始化view的各控件
     isPrepared =  true ;
         lazyLoad();
         return  view;
     }
     @Override
     protected void lazyLoad() {
         if (!isPrepared || !isVisible) {
             return ;
         }
         //填充各控件的数据
     }
}

在上面的类当中,我们增加了一个标志位isPrepared,用于标志是否初始化完成。然后在我们所需要的初始化操作完成之后调用,如上面的例子当中,在初始化view之后,设置 isPrepared为true,同时调用lazyLoad()方法。而在lazyLoad()当中,判断isPrepared和isVisible只要有一个不为true就不往下执行。也就是仅当初始化完成,并且可见的时候才继续加载,这样的避免了未初始化完成就使用而带来的问题。

在这里我对fragment的懒加载实现的介绍就到此为止,如果你有兴趣,可以基于此再深入探究,比如写一个带有缓初始化和可见时刷新的特性的Fragment。

  1. public void setOffscreenPageLimit(int limit) {  
  2.         if (limit < DEFAULT_OFFSCREEN_PAGES) {  
  3.             Log.w(TAG, "Requested offscreen page limit " + limit + " too small; defaulting to " +  
  4.                     DEFAULT_OFFSCREEN_PAGES);  
  5.             limit = DEFAULT_OFFSCREEN_PAGES;  
  6.         }  
  7.         if (limit != mOffscreenPageLimit) {  
  8.             mOffscreenPageLimit = limit;  
  9.             populate();  
  10.         }  
  11.     }  
  12.   
  13. 从这个方法来看,不管你设置什么值,至少会预先加载下一个Fragment,你想预先加载几个就可以传入相应的参数.  
  14. 这种情况如音乐播放时,如果有自动加载歌词就可以使用了.  
  15.   
  16. 如果你的界面需要加载一些大量的数据,但你不想预先加载下一个界面(需要网络或耗时的操作),使用ViewPager却很无耐.特别是下一个界面有可能你一段很长时间不会使用到,如我开发的微博,在显示主页后我不想立即加载下一个界面,因为都有ListView,如果我不访问它,就不必加载无用的资源.  
  17.   
  18. 可以通过修改这个值,但有,修改后就会有一个麻烦的地方,因为移动时不会预先加载下一个界面的关系,所以会看到一片黑色的背景.  
  19.   
  20. 如果不介意黑色背景,可以覆盖这个类,然后定义默认的加载数量为0  
  21. private int mOffscreenPageLimit = DEFAULT_OFFSCREEN_PAGES=0;就是不预先加载下一个界面.  
  22.   
  23. 如果想预加载,可以使用原来的ViewPager,或这里直接改为mOffscreenPageLimit=你要加载的数量。  
  24.   
  25. 由于sdk的更新问题,不是所有v4 support都适合的,v17时就换了  
  26.   
  27. 先到这吧.  

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值