在项目中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(),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。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持编程小技巧。
总结以上是内存溢出为你收集整理的android实现ViewPager懒加载的三种方法全部内容,希望文章能够帮你解决android实现ViewPager懒加载的三种方法所遇到的程序开发问题。
如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)