欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

android实现ViewPager懒加载的三种方法

程序员文章站 2023-12-14 11:26:58
在项目中viewpager和fragment接口框架已经是处处可见,但是在使用中,我们肯定不希望用户在当前页面时就在前后页面的数据,加入数据量很大,而用户又不愿意左右滑动浏...

在项目中viewpager和fragment接口框架已经是处处可见,但是在使用中,我们肯定不希望用户在当前页面时就在前后页面的数据,加入数据量很大,而用户又不愿意左右滑动浏览,那么这时候viewpager中本来充满善意的预加载就有点令人不爽了。我们能做的就是屏蔽掉viewpager的预加载机制。虽然viewpager中提供的有setoffscreenpagelimit()来控制其预加载的数目,但是当设置为0后我们发现其根本没效果,这个的最小值就是1,也就是你只能最少前后各预加载一页。那么,这时候就得另觅方法了。

以下三种方法是我在学习和项目中尝试过的,需求实现了,但各有千秋,可结合不同场景使用。因为打算慢慢养成写博客的习惯,就总结在此,也希望对他人有所借鉴。

方法一

在fragment可见时请求数据。此方案仍预加载了前后的页面,但是没有请求数据,只有进入到当前framgent时才请求数据。

优点:实现了数据的懒加载

缺点:一次仍是三个framgment对象,不是完全意义的懒加载

public class fragmentsample extends fragment{
  ... 
  @override
  public void setuservisiblehint(boolean isvisibletouser) {
    super.setuservisiblehint(isvisibletouser);
    if (isvisibletouser) {
      requestdata(); // 在此请求数据
    }
  }
  ...
}

方法二

直接修改viewpager源码。通过查看viewpager源码可知,控制其预加载的是一个常量default_offscreen_pages,其默认值为1,表示当前页面前后各预加载一个页面,在这里我们直接将其设置为0即可,即去掉预加载。但是,这样有一个问题,那就是在使用其他控件时需要传入viewpager时,这个就不能用了。

优点:完全屏蔽掉了预加载

缺点:应用太受限制,比如使用viewpagerindicator时需要传入viewpager对象,这时傻眼了。

// 注意,这是直接拷贝的viewpager的源码,只修改了注释处的代码
public class lazyviewpager extends viewgroup {
 private static final string tag = "lazyviewpager";
 private static final boolean debug = false;
 private static final boolean use_cache = false;
   // 默认为1,即前后各预加载一个页面,设置为0去掉预加载
   private static final int default_offscreen_pages = 0;
 private static final int max_settle_duration = 600; // ms
 static class iteminfo {
 object object;
 int position;
 boolean scrolling;
 }
 private static final comparator<iteminfo> comparator = new comparator<iteminfo>() {
 @override
 public int compare(iteminfo lhs, iteminfo rhs) {
  return lhs.position - rhs.position;
 }
 };
   ............
}

方法三

直接继承viewpager,结合pageradapter实现懒加载。该方案是我用到的最完善的方法,完全的懒加载,每次只会建立一个fragment对象。

优点:完全屏蔽预加载

缺点:稍微复杂,但是人家已经造好的*,直接用吧,很简洁

代码下载:lazyviewpager_jb51.rar

这个库就4个类,作者通过继承viewpager(保证其普适性)、自定义viewpageradapter和 lazyfragmentpageradapter以及设置懒加载的标记接口,很好的实现了懒加载。感谢作者。

在此贴出关键代码,有兴趣的同学可以学习下。

lazyviewpager:

public class lazyviewpager extends viewpager {
 private static final float default_offset = 0.5f;
 private lazypageradapter mlazypageradapter;
 private float minitlazyitemoffset = default_offset;
 public lazyviewpager(context context) {
 super(context);
 }
 public lazyviewpager(context context, attributeset attrs) {
 super(context, attrs);
 typedarray a = context.obtainstyledattributes(attrs, r.styleable.lazyviewpager);
 setinitlazyitemoffset(a.getfloat(r.styleable.lazyviewpager_init_lazy_item_offset, default_offset));
 a.recycle();
 }
  /**
   * change the initlazyitemoffset
   * @param initlazyitemoffset set minitlazyitemoffset if {@code 0 < initlazyitemoffset <= 1}
   */
 public void setinitlazyitemoffset(float initlazyitemoffset) {
 if (initlazyitemoffset > 0 && initlazyitemoffset <= 1) {
   minitlazyitemoffset = initlazyitemoffset;
    }
 }
 @override
 public void setadapter(pageradapter adapter) {
 super.setadapter(adapter);
    mlazypageradapter = adapter != null && adapter instanceof lazypageradapter ? (lazypageradapter) adapter : null;
 }
 @override
 protected void onpagescrolled(int position, float offset, int offsetpixels) {
 if (mlazypageradapter != null) {
  if (getcurrentitem() == position) {
  int lazyposition = position + 1;
  if (offset >= minitlazyitemoffset && mlazypageradapter.islazyitem(lazyposition)) {
          mlazypageradapter.startupdate(this);
          mlazypageradapter.addlazyitem(this, lazyposition);
          mlazypageradapter.finishupdate(this);
  }
  } else if (getcurrentitem() > position) {
  int lazyposition = position;
  if (1 - offset >= minitlazyitemoffset && mlazypageradapter.islazyitem(lazyposition)) {
          mlazypageradapter.startupdate(this);
          mlazypageradapter.addlazyitem(this, lazyposition);
          mlazypageradapter.finishupdate(this);
  }
  }
 }
 super.onpagescrolled(position, offset, offsetpixels);
 }
}
public abstract class lazyfragmentpageradapter extends lazypageradapter<fragment> {
 private static final string tag = "lazyfragmentpageradapter";
 private static final boolean debug = false;
 private final fragmentmanager mfragmentmanager;
 private fragmenttransaction mcurtransaction = null;
 public lazyfragmentpageradapter(fragmentmanager fm) {
 mfragmentmanager = fm;
 }
 @override
 public void startupdate(viewgroup container) {
 }
 @override
 public object instantiateitem(viewgroup container, int position) {
 if (mcurtransaction == null) {
  mcurtransaction = mfragmentmanager.begintransaction();
 }
 final long itemid = getitemid(position);
 // do we already have this fragment?
 string name = makefragmentname(container.getid(), itemid);
 fragment fragment = mfragmentmanager.findfragmentbytag(name);
 if (fragment != null) {
  if (debug)
  log.v(tag, "attaching item #" + itemid + ": f=" + fragment);
  mcurtransaction.attach(fragment);
 } else {
  fragment = getitem(container, position);
  if (fragment instanceof laziable) {
  mlazyitems.put(position, fragment);
  } else {
  mcurtransaction.add(container.getid(), fragment, name);
  }
 }
 if (fragment != getcurrentitem()) {
  fragment.setmenuvisibility(false);
  fragment.setuservisiblehint(false);
 }
 return fragment;
 }
 @override
 public void destroyitem(viewgroup container, int position, object object) {
 if (mcurtransaction == null) {
  mcurtransaction = mfragmentmanager.begintransaction();
 }
 if (debug)
  log.v(tag, "detaching item #" + getitemid(position) + ": f=" + object + " v=" + ((fragment) object).getview());
 final long itemid = getitemid(position);
 string name = makefragmentname(container.getid(), itemid);
 if (mfragmentmanager.findfragmentbytag(name) == null) {
  mcurtransaction.detach((fragment) object);
 } else {
      mlazyitems.remove(position);
 }
 }
  @override
 public fragment addlazyitem(viewgroup container, int position) {
 fragment fragment = mlazyitems.get(position);
 if (fragment == null)
  return null;
 final long itemid = getitemid(position);
 string name = makefragmentname(container.getid(), itemid);
 if (mfragmentmanager.findfragmentbytag(name) == null) {
  if (mcurtransaction == null) {
  mcurtransaction = mfragmentmanager.begintransaction();
  }
  mcurtransaction.add(container.getid(), fragment, name);
      mlazyitems.remove(position);
 }
    return fragment;
 }
 @override
 public void finishupdate(viewgroup container) {
 if (mcurtransaction != null) {
  mcurtransaction.commitallowingstateloss();
  mcurtransaction = null;
  mfragmentmanager.executependingtransactions();
 }
 }
  @override
 public boolean isviewfromobject(view view, object object) {
 return ((fragment) object).getview() == view;
 }
 public long getitemid(int position) {
 return position;
 }
 private static string makefragmentname(int viewid, long id) {
 return "android:switcher:" + viewid + ":" + id;
 }
  /**
   * mark the fragment can be added lazily
   */
  public interface laziable {
  }
}

最后提醒一下:填充lazyviewpager的fragment一定要实现接口lazyfragmentpageradapter.laziable。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。

上一篇:

下一篇: