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

以下三种方法各有千秋,可结合不同场景使用。

方法一

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

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

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

  1. public class FragmentSample extends Fragment{ @Override
  2. public void setUserVisibleHint(boolean isVisibleToUser) {
  3. super.setUserVisibleHint(isVisibleToUser);
  4. if (isVisibleToUser) {
  5. requestData(); // 在此请求数据
  6. }
  7. }
  8. ...
  9. }

方法二

直接修改ViewPager源码。通过查看ViewPager源码可知,控制其预加载的是一个常量

DEFAULT_OFFSCREEN_PAGES,其默认值为1,表示当前页面前后各预加载一个页面,在这里我们直接将其设置为0即可,即去掉预加载。但是,这样有一个问题,那就是在使用其他控件时需要传入ViewPager时,这个就不能用了。

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

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

  1. // 注意,这是直接拷贝的ViewPager的源码,只修改了注释处的代码
  2. 2.public class LazyViewPager extends ViewGroup {
  3. 3. private static final String TAG = "LazyViewPager";
  4. 4. private static final boolean DEBUG = false;
  5. 5.
  6. 6. private static final boolean USE_CACHE = false;
  7. 7.
  8. 8. // 默认为1,即前后各预加载一个页面,设置为0去掉预加载
  9. 9. private static final int DEFAULT_OFFSCREEN_PAGES = 0;
  10. 10.
  11. 11. private static final int MAX_SETTLE_DURATION = 600; // ms
  12. 12.
  13. 13. static class ItemInfo {
  14. 14. Object object;
  15. 15. int position;
  16. 16. boolean scrolling;
  17. 17. }
  18. 18.
  19. 19. private static final Comparator<ItemInfo> COMPARATOR = new Comparator<ItemInfo>() {
  20. 20. @Override
  21. 21. public int compare(ItemInfo lhs, ItemInfo rhs) {
  22. 22. return lhs.position - rhs.position;
  23. 23. }
  24. 24. };
  25. 25. ............
  26. 26.}

方法三

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

优点:完全屏蔽预加载

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

开源库:https://github.com/lianghanzhen/LazyViewPager

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

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

LazyViewPager:

  1. public class LazyViewPager extends ViewPager {
  2. 2.
  3. 3. private static final float DEFAULT_OFFSET = 0.5f;
  4. 4.
  5. 5. private LazyPagerAdapter mLazyPagerAdapter;
  6. 6. private float mInitLazyItemOffset = DEFAULT_OFFSET;
  7. 7.
  8. 8. public LazyViewPager(Context context) {
  9. 9. super(context);
  10. 10. }
  11. 11.
  12. 12. public LazyViewPager(Context context, AttributeSet attrs) {
  13. 13. super(context, attrs);
  14. 14.
  15. 15. TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.LazyViewPager);
  16. 16. setInitLazyItemOffset(a.getFloat(R.styleable.LazyViewPager_init_lazy_item_offset, DEFAULT_OFFSET));
  17. 17. a.recycle();
  18. 18. }
  19. 19.
  20. 20. /**
  21. 21. * change the initLazyItemOffset
  22. 22. * @param initLazyItemOffset set mInitLazyItemOffset if {@code 0 < initLazyItemOffset <= 1}
  23. 23. */
  24. 24. public void setInitLazyItemOffset(float initLazyItemOffset) {
  25. 25. if (initLazyItemOffset > 0 && initLazyItemOffset <= 1) {
  26. 26. mInitLazyItemOffset = initLazyItemOffset;
  27. 27. }
  28. 28. }
  29. 29.
  30. 30. @Override
  31. 31. public void setAdapter(PagerAdapter adapter) {
  32. 32. super.setAdapter(adapter);
  33. 33. mLazyPagerAdapter = adapter != null && adapter instanceof LazyPagerAdapter ? (LazyPagerAdapter) adapter : null;
  34. 34. }
  35. 35.
  36. 36. @Override
  37. 37. protected void onPageScrolled(int position, float offset, int offsetPixels) {
  38. 38. if (mLazyPagerAdapter != null) {
  39. 39. if (getCurrentItem() == position) {
  40. 40. int lazyPosition = position + 1;
  41. 41. if (offset >= mInitLazyItemOffset && mLazyPagerAdapter.isLazyItem(lazyPosition)) {
  42. 42. mLazyPagerAdapter.startUpdate(this);
  43. 43. mLazyPagerAdapter.addLazyItem(this, lazyPosition);
  44. 44. mLazyPagerAdapter.finishUpdate(this);
  45. 45. }
  46. 46. } else if (getCurrentItem() > position) {
  47. 47. int lazyPosition = position;
  48. 48. if (1 - offset >= mInitLazyItemOffset && mLazyPagerAdapter.isLazyItem(lazyPosition)) {
  49. 49. mLazyPagerAdapter.startUpdate(this);
  50. 50. mLazyPagerAdapter.addLazyItem(this, lazyPosition);
  51. 51. mLazyPagerAdapter.finishUpdate(this);
  52. 52. }
  53. 53. }
  54. 54. }
  55. 55. super.onPageScrolled(position, offset, offsetPixels);
  56. 56. }
  57. 57}
  1. public abstract class LazyFragmentPagerAdapter extends LazyPagerAdapter<Fragment> {
  2. 2.
  3. 3. private static final String TAG = "LazyFragmentPagerAdapter";
  4. 4. private static final boolean DEBUG = false;
  5. 5.
  6. 6. private final FragmentManager mFragmentManager;
  7. 7. private FragmentTransaction mCurTransaction = null;
  8. 8.
  9. 9. public LazyFragmentPagerAdapter(FragmentManager fm) {
  10. 10. mFragmentManager = fm;
  11. 11. }
  12. 12.
  13. 13. @Override
  14. 14. public void startUpdate(ViewGroup container) {
  15. 15. }
  16. 16.
  17. 17. @Override
  18. 18. public Object instantiateItem(ViewGroup container, int position) {
  19. 19. if (mCurTransaction == null) {
  20. 20. mCurTransaction = mFragmentManager.beginTransaction();
  21. 21. }
  22. 22.
  23. 23. final long itemId = getItemId(position);
  24. 24.
  25. 25. // Do we already have this fragment?
  26. 26. String name = makeFragmentName(container.getId(), itemId);
  27. 27. Fragment fragment = mFragmentManager.findFragmentByTag(name);
  28. 28. if (fragment != null) {
  29. 29. if (DEBUG)
  30. 30. Log.v(TAG, "Attaching item #" + itemId + ": f=" + fragment);
  31. 31. mCurTransaction.attach(fragment);
  32. 32. } else {
  33. 33. fragment = getItem(container, position);
  34. 34. if (fragment instanceof Laziable) {
  35. 35. mLazyItems.put(position, fragment);
  36. 36. } else {
  37. 37. mCurTransaction.add(container.getId(), fragment, name);
  38. 38. }
  39. 39. }
  40. 40. if (fragment != getCurrentItem()) {
  41. 41. fragment.setMenuVisibility(false);
  42. 42. fragment.setUserVisibleHint(false);
  43. 43. }
  44. 44.
  45. 45. return fragment;
  46. 46. }
  47. 47.
  48. 48. @Override
  49. 49. public void destroyItem(ViewGroup container, int position, Object object) {
  50. 50. if (mCurTransaction == null) {
  51. 51. mCurTransaction = mFragmentManager.beginTransaction();
  52. 52. }
  53. 53. if (DEBUG)
  54. 54. Log.v(TAG, "Detaching item #" + getItemId(position) + ": f=" + object + " v=" + ((Fragment) object).getView());
  55. 55.
  56. 56. final long itemId = getItemId(position);
  57. 57. String name = makeFragmentName(container.getId(), itemId);
  58. 58. if (mFragmentManager.findFragmentByTag(name) == null) {
  59. 59. mCurTransaction.detach((Fragment) object);
  60. 60. } else {
  61. 61. mLazyItems.remove(position);
  62. 62. }
  63. 63. }
  64. 64.
  65. 65. @Override
  66. 66. public Fragment addLazyItem(ViewGroup container, int position) {
  67. 67. Fragment fragment = mLazyItems.get(position);
  68. 68. if (fragment == null)
  69. 69. return null;
  70. 70.
  71. 71. final long itemId = getItemId(position);
  72. 72. String name = makeFragmentName(container.getId(), itemId);
  73. 73. if (mFragmentManager.findFragmentByTag(name) == null) {
  74. 74. if (mCurTransaction == null) {
  75. 75. mCurTransaction = mFragmentManager.beginTransaction();
  76. 76. }
  77. 77. mCurTransaction.add(container.getId(), fragment, name);
  78. 78. mLazyItems.remove(position);
  79. 79. }
  80. 80. return fragment;
  81. 81. }
  82. 82.
  83. 83. @Override
  84. 84. public void finishUpdate(ViewGroup container) {
  85. 85. if (mCurTransaction != null) {
  86. 86. mCurTransaction.commitAllowingStateLoss();
  87. 87. mCurTransaction = null;
  88. 88. mFragmentManager.executePendingTransactions();
  89. 89. }
  90. 90. }
  91. 91.
  92. 92. @Override
  93. 93. public boolean isViewFromObject(View view, Object object) {
  94. 94. return ((Fragment) object).getView() == view;
  95. 95. }
  96. 96.
  97. 97. public long getItemId(int position) {
  98. 98. return position;
  99. 99. }
  100. 100.
  101. 101. private static String makeFragmentName(int viewId, long id) {
  102. 102. return "android:switcher:" + viewId + ":" + id;
  103. 103. }
  104. 104.
  105. 105. /**
  106. 106. * mark the fragment can be added lazily
  107. 107. */
  108. 108. public interface Laziable {
  109. 109. }
  110. 110.
  111. 111.}

最后提醒一下:填充LazyViewPager的Fragment一定要实现接口LazyFragmentPagerAdapter.Laziable。

=========================================================================================

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

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

  1. public abstract class LazyFragment extends Fragment {
  2. 12. protected boolean isVisible;
  3. 13. /**
  4. 14. * 在这里实现Fragment数据的缓加载.
  5. 15. * @param isVisibleToUser
  6. 16. */
  7. 17. @Override
  8. 18. public void setUserVisibleHint(boolean isVisibleToUser) {
  9. 19. super.setUserVisibleHint(isVisibleToUser);
  10. 20. if(getUserVisibleHint()) {
  11. 21. isVisible = true;
  12. 22. onVisible();
  13. 23. } else {
  14. 24. isVisible = false;
  15. 25. onInvisible();
  16. 26. }
  17. 27. }
  18. 28. protected void onVisible(){
  19. 29. lazyLoad();
  20. 30. }
  21. 31. protected abstract void lazyLoad();
  22. 32. protected void onInvisible(){}
  23. 33.}

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

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

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

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

【转】ViewPager学习笔记(一)——懒加载的更多相关文章

  1. thinkphp学习笔记9—自动加载

    原文:thinkphp学习笔记9-自动加载 1.命名空间自动加载 在3.2版本中不需要手动加载类库文件,可以很方便的完成自动加载. 系统可以根据类的命名空间自动定位到类库文件,例如定义了一个类Org\ ...

  2. node 学习笔记 - Modules 模块加载系统 (1)

    本文同步自我的个人博客:http://www.52cik.com/2015/12/11/learn-node-modules-path.html 用了这么久的 require,但却没有系统的学习过 n ...

  3. Openstack本学习笔记——Neutron-server服务加载和启动源代码分析(三)

    本文是在学习Openstack过程中整理和总结.因为时间和个人能力有限.错误之处在所难免,欢迎指正! 在Neutron-server服务载入与启动源代码分析(二)中搞定模块功能的扩展和载入.我们就回到 ...

  4. Spring学习笔记(1)——资源加载

    <!-- 占坑,迟点补充底层原理 --> Spring支持4种资源的地址前缀 (1)从类路径中加载资源——classpath: classpath:和classpath:/是等价的,都是相 ...

  5. node 学习笔记 - Modules 模块加载系统 (2)

    本文同步自我的个人博客:http://www.52cik.com/2015/12/14/learn-node-modules-module.html 上一篇讲了模块是如何被寻找到然后加载进来的,这篇则 ...

  6. 【EF学习笔记08】----------加载关联表的数据 显式加载

    显式加载 讲解之前,先来看一下我们的数据库结构:班级表 学生表 加载从表集合类型 //显示加载 Console.WriteLine("=========查询集合===========&quo ...

  7. 【EF学习笔记07】----------加载关联表的数据 贪婪加载

    讲解之前,先来看一下我们的数据库结构:班级表 学生表 贪婪加载 //贪婪加载 using (var db = new Entities()) { var classes = db.Classes.Wh ...

  8. 【EF学习笔记06】----------加载关联表的数据 延迟加载

    讲解之前,先来看一下我们的数据库结构:班级表 学生表 延迟加载 //延迟加载 using (var db = new Entities()) { //查询班级 var classes = (from ...

  9. webpack学习笔记--按需加载

    为什么需要按需加载 随着互联网的发展,一个网页需要承载的功能越来越多. 对于采用单页应用作为前端架构的网站来说,会面临着一个网页需要加载的代码量很大的问题,因为许多功能都集中的做到了一个 HTML 里 ...

  10. Office365学习笔记—Lookup类型加载条目过多解决方案

    1,随着接触的项目越来越多,遇到的各种奇葩的问题也越来越多,不得不说,SharePoint是个好东西,提高了开发效率,简化了很多基础的功能.但是令人头疼的问题是,当你想做个稍微复杂点的功能,就不得不研 ...

随机推荐

  1. java(搜索不区分大小写)

    ref.put("myfield", Pattern.compile(".*myValue.*" , Pattern.CASE_INSENSITIVE));

  2. OC中的指针

    NSError *err = nil; NSError __strong **error = &err; //因为在oc中,通过* *err 创建的指针是用__strong修改的,所以要一致, ...

  3. 百度搜索--jquery

    <!doctype html> <html> <head> <meta charset="utf-8"> <title> ...

  4. window10 安装出现the error code is 2503错误的解决方法

    window10 安装出现the error code is 2503错误的解决方法:  设置 C:\WINDOWS\TEMP的权限

  5. Java 日志性能优化

    1. 选择合理的日志级别.合理控制日志内容 2. 控制日志的输出内容和格式 logger.debug("Entry number: " + i + " is " ...

  6. Broadmann分区

    来源: http://blog.sina.com.cn/s/blog_60a751620100k2hj.html Brodmann areas Name 中文名 Function 1 Somatose ...

  7. MYSQL临时表创建索引

    DROP TEMPORARY TABLE IF EXISTS tmp_record_t2;CREATE TEMPORARY TABLE tmp_record_t2(consumption_id INT ...

  8. 投入Html5的怀抱,最近在研究的Egret

    html5没有办法不关注,实在太火热了,几年前还不行,如今确是环境较好,typescript语言很好学习,可能基于之前的基础,不到一个星期就基本上差不多了,虽然还有一些小问题,但那都是经验积累下来可以 ...

  9. Hangfire项目实践

    Hangfire项目实践分享 Hangfire项目实践分享 目录 Hangfire项目实践分享 目录 什么是Hangfire Hangfire基础 基于队列的任务处理(Fire-and-forget ...

  10. 在iframe中获取本iframe DOM引用

    window.frameElement 获取本iframe DOM window.frameElement.contentDocument.getElementById('id') 获取这个ifram ...