android实现ViewPager懒加载的三种方法
在项目中ViewPager和Fragment接口框架已经是处处可见,但是在使用中,我们肯定不希望用户在当前页面时就在前后页面的数据,加入数据量很大,而用户又不愿意左右滑动浏览,那么这时候ViewPager中本来充满善意的预加载就有点令人不爽了。我们能做的就是屏蔽掉ViewPager的预加载机制。虽然ViewPager中提供的有setOffscreenPageLimit()来控制其预加载的数目,但是当设置为0后我们发现其根本没效果,这个的最小值就是1,也就是你只能最少前后各预加载一页。那么,这时候就得另觅方法了。
以下三种方法是我在学习和项目中尝试过的,需求实现了,但各有千秋,可结合不同场景使用。因为打算慢慢养成写博客的习惯,就总结在此,也希望对他人有所借鉴。
方法一
在Fragment可见时请求数据。此方案仍预加载了前后的页面,但是没有请求数据,只有进入到当前Framgent时才请求数据。
优点:实现了数据的懒加载
缺点:一次仍是三个Framgment对象,不是完全意义的懒加载
publicclassFragmentSampleextendsFragment{
...
@Override
publicvoidsetUserVisibleHint(booleanisVisibleToUser){
super.setUserVisibleHint(isVisibleToUser);
if(isVisibleToUser){
requestData();//在此请求数据
}
}
...
}
方法二
直接修改ViewPager源码。通过查看ViewPager源码可知,控制其预加载的是一个常量DEFAULT_OFFSCREEN_PAGES,其默认值为1,表示当前页面前后各预加载一个页面,在这里我们直接将其设置为0即可,即去掉预加载。但是,这样有一个问题,那就是在使用其他控件时需要传入ViewPager时,这个就不能用了。
优点:完全屏蔽掉了预加载
缺点:应用太受限制,比如使用ViewPagerIndicator时需要传入ViewPager对象,这时傻眼了。
//注意,这是直接拷贝的ViewPager的源码,只修改了注释处的代码
publicclassLazyViewPagerextendsViewGroup{
privatestaticfinalStringTAG="LazyViewPager";
privatestaticfinalbooleanDEBUG=false;
privatestaticfinalbooleanUSE_CACHE=false;
//默认为1,即前后各预加载一个页面,设置为0去掉预加载
privatestaticfinalintDEFAULT_OFFSCREEN_PAGES=0;
privatestaticfinalintMAX_SETTLE_DURATION=600;//ms
staticclassItemInfo{
Objectobject;
intposition;
booleanscrolling;
}
privatestaticfinalComparatorCOMPARATOR=newComparator(){
@Override
publicintcompare(ItemInfolhs,ItemInforhs){
returnlhs.position-rhs.position;
}
};
............
}
方法三
直接继承ViewPager,结合PagerAdapter实现懒加载。该方案是我用到的最完善的方法,完全的懒加载,每次只会建立一个Fragment对象。
优点:完全屏蔽预加载
缺点:稍微复杂,但是人家已经造好的轮子,直接用吧,很简洁
代码下载:LazyViewPager_jb51.rar
这个库就4个类,作者通过继承ViewPager(保证其普适性)、自定义ViewPagerAdapter和LazyFragmentPagerAdapter以及设置懒加载的标记接口,很好的实现了懒加载。感谢作者。
在此贴出关键代码,有兴趣的同学可以学习下。
LazyViewPager:
publicclassLazyViewPagerextendsViewPager{
privatestaticfinalfloatDEFAULT_OFFSET=0.5f;
privateLazyPagerAdaptermLazyPagerAdapter;
privatefloatmInitLazyItemOffset=DEFAULT_OFFSET;
publicLazyViewPager(Contextcontext){
super(context);
}
publicLazyViewPager(Contextcontext,AttributeSetattrs){
super(context,attrs);
TypedArraya=context.obtainStyledAttributes(attrs,R.styleable.LazyViewPager);
setInitLazyItemOffset(a.getFloat(R.styleable.LazyViewPager_init_lazy_item_offset,DEFAULT_OFFSET));
a.recycle();
}
/**
*changetheinitLazyItemOffset
*@paraminitLazyItemOffsetsetmInitLazyItemOffsetif{@code00&&initLazyItemOffset<=1){
mInitLazyItemOffset=initLazyItemOffset;
}
}
@Override
publicvoidsetAdapter(PagerAdapteradapter){
super.setAdapter(adapter);
mLazyPagerAdapter=adapter!=null&&adapterinstanceofLazyPagerAdapter?(LazyPagerAdapter)adapter:null;
}
@Override
protectedvoidonPageScrolled(intposition,floatoffset,intoffsetPixels){
if(mLazyPagerAdapter!=null){
if(getCurrentItem()==position){
intlazyPosition=position+1;
if(offset>=mInitLazyItemOffset&&mLazyPagerAdapter.isLazyItem(lazyPosition)){
mLazyPagerAdapter.startUpdate(this);
mLazyPagerAdapter.addLazyItem(this,lazyPosition);
mLazyPagerAdapter.finishUpdate(this);
}
}elseif(getCurrentItem()>position){
intlazyPosition=position;
if(1-offset>=mInitLazyItemOffset&&mLazyPagerAdapter.isLazyItem(lazyPosition)){
mLazyPagerAdapter.startUpdate(this);
mLazyPagerAdapter.addLazyItem(this,lazyPosition);
mLazyPagerAdapter.finishUpdate(this);
}
}
}
super.onPageScrolled(position,offset,offsetPixels);
}
}
publicabstractclassLazyFragmentPagerAdapterextendsLazyPagerAdapter{ privatestaticfinalStringTAG="LazyFragmentPagerAdapter"; privatestaticfinalbooleanDEBUG=false; privatefinalFragmentManagermFragmentManager; privateFragmentTransactionmCurTransaction=null; publicLazyFragmentPagerAdapter(FragmentManagerfm){ mFragmentManager=fm; } @Override publicvoidstartUpdate(ViewGroupcontainer){ } @Override publicObjectinstantiateItem(ViewGroupcontainer,intposition){ if(mCurTransaction==null){ mCurTransaction=mFragmentManager.beginTransaction(); } finallongitemId=getItemId(position); //Dowealreadyhavethisfragment? Stringname=makeFragmentName(container.getId(),itemId); Fragmentfragment=mFragmentManager.findFragmentByTag(name); if(fragment!=null){ if(DEBUG) Log.v(TAG,"Attachingitem#"+itemId+":f="+fragment); mCurTransaction.attach(fragment); }else{ fragment=getItem(container,position); if(fragmentinstanceofLaziable){ mLazyItems.put(position,fragment); }else{ mCurTransaction.add(container.getId(),fragment,name); } } if(fragment!=getCurrentItem()){ fragment.setMenuVisibility(false); fragment.setUserVisibleHint(false); } returnfragment; } @Override publicvoiddestroyItem(ViewGroupcontainer,intposition,Objectobject){ if(mCurTransaction==null){ mCurTransaction=mFragmentManager.beginTransaction(); } if(DEBUG) Log.v(TAG,"Detachingitem#"+getItemId(position)+":f="+object+"v="+((Fragment)object).getView()); finallongitemId=getItemId(position); Stringname=makeFragmentName(container.getId(),itemId); if(mFragmentManager.findFragmentByTag(name)==null){ mCurTransaction.detach((Fragment)object); }else{ mLazyItems.remove(position); } } @Override publicFragmentaddLazyItem(ViewGroupcontainer,intposition){ Fragmentfragment=mLazyItems.get(position); if(fragment==null) returnnull; finallongitemId=getItemId(position); Stringname=makeFragmentName(container.getId(),itemId); if(mFragmentManager.findFragmentByTag(name)==null){ if(mCurTransaction==null){ mCurTransaction=mFragmentManager.beginTransaction(); } mCurTransaction.add(container.getId(),fragment,name); mLazyItems.remove(position); } returnfragment; } @Override publicvoidfinishUpdate(ViewGroupcontainer){ if(mCurTransaction!=null){ mCurTransaction.commitAllowingStateLoss(); mCurTransaction=null; mFragmentManager.executePendingTransactions(); } } @Override publicbooleanisViewFromObject(Viewview,Objectobject){ return((Fragment)object).getView()==view; } publiclonggetItemId(intposition){ returnposition; } privatestaticStringmakeFragmentName(intviewId,longid){ return"android:switcher:"+viewId+":"+id; } /** *markthefragmentcanbeaddedlazily */ publicinterfaceLaziable{ } }
最后提醒一下:填充LazyViewPager的Fragment一定要实现接口LazyFragmentPagerAdapter.Laziable。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持毛票票。