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

发布时间 - 2026-01-10 23:28:23    点击率:

在项目中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。

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


# viewpager实现懒加载  # viewpager懒加载  # viewpager的懒加载  # Android之Viewpager+Fragment实现懒加载示例  # Android ViewPager动态加载问题  # 详解Android_性能优化之ViewPager加载成百上千高清大图oom解决方案  # android 解决ViewPager加载大量图片内存溢出问题  # Android ViewPager制作新手导航页(动态加载)  # Android 使用ViewPager自动滚动循环轮播效果  # Android ViewPager实现图片轮播效果  # Android使用ViewPager实现自动轮播  # Android中用RxJava和ViewPager实现轮播图  # Android使用ViewPager加载图片和轮播视频  # 加载  # 在此  # 设置为  # 实现了  # 时需  # 的是  # 这是  # 这时候  # 是在  # 在这里  # 很好  # 我在  # 那就  # 各有千秋  # 将其  # 只会  # 用了  # 仍是  # 有兴趣  # 三种 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 电商网站制作价格怎么算,网上拍卖流程以及规则?  黑客如何利用漏洞与弱口令入侵网站服务器?  C#如何调用原生C++ COM对象详解  如何快速生成ASP一键建站模板并优化安全性?  php8.4header发送头信息失败怎么办_php8.4header函数问题解决【解答】  php做exe能调用系统命令吗_执行cmd指令实现方式【详解】  JS去除重复并统计数量的实现方法  实例解析angularjs的filter过滤器  Microsoft Edge如何解决网页加载问题 Edge浏览器加载问题修复  Laravel如何使用Vite进行前端资源打包?(配置示例)  如何用搬瓦工VPS快速搭建个人网站?  Laravel Octane如何提升性能_使用Laravel Octane加速你的应用  制作公司内部网站有哪些,内网如何建网站?  宙斯浏览器文件分类查看教程 快速筛选视频文档与图片方法  JavaScript 输出显示内容(document.write、alert、innerHTML、console.log)  湖南网站制作公司,湖南上善若水科技有限公司做什么的?  Laravel怎么为数据库表字段添加索引以优化查询  Laravel如何正确地在控制器和模型之间分配逻辑_Laravel代码职责分离与架构建议  Laravel如何理解并使用服务容器(Service Container)_Laravel依赖注入与容器绑定说明  使用PHP下载CSS文件中的所有图片【几行代码即可实现】  如何用美橙互联一键搭建多站合一网站?  微信小程序 配置文件详细介绍  黑客入侵网站服务器的常见手法有哪些?  如何快速生成橙子建站落地页链接?  如何在万网自助建站平台快速创建网站?  北京企业网站设计制作公司,北京铁路集团官方网站?  香港服务器建站指南:免备案优势与SEO优化技巧全解析  如何快速上传建站程序避免常见错误?  JavaScript如何实现倒计时_时间函数如何精确控制  Laravel storage目录权限问题_Laravel文件写入权限设置  Python图片处理进阶教程_Pillow滤镜与图像增强  ChatGPT 4.0官网入口地址 ChatGPT在线体验官网  长沙做网站要多少钱,长沙国安网络怎么样?  在线制作视频的网站有哪些,电脑如何制作视频短片?  Android中AutoCompleteTextView自动提示  Laravel如何从数据库删除数据_Laravel destroy和delete方法区别  Laravel如何配置和使用缓存?(Redis代码示例)  历史网站制作软件,华为如何找回被删除的网站?  HTML5建模怎么导出为FBX格式_FBX格式兼容性及导出步骤【指南】  Swift开发中switch语句值绑定模式  DeepSeek是免费使用的吗 DeepSeek收费模式与Pro版本功能详解  公司网站制作价格怎么算,公司办个官网需要多少钱?  WordPress 子目录安装中正确处理脚本路径的完整指南  mc皮肤壁纸制作器,苹果平板怎么设置自己想要的壁纸我的世界?  EditPlus中的正则表达式 实战(1)  免费制作统计图的网站有哪些,如何看待现如今年轻人买房难的情况?  edge浏览器无法安装扩展 edge浏览器插件安装失败【解决方法】  html5怎么画眼睛_HT5用Canvas或SVG画眼球瞳孔加JS控制动态【绘制】  Laravel如何升级到最新的版本_Laravel版本升级流程与兼容性处理  javascript事件捕获机制【深入分析IE和DOM中的事件模型】