android實(shí)現(xiàn)ViewPager懶加載的三種方法
在項(xiàng)目中ViewPager和Fragment接口框架已經(jīng)是處處可見(jiàn),但是在使用中,我們肯定不希望用戶在當(dāng)前頁(yè)面時(shí)就在前后頁(yè)面的數(shù)據(jù),加入數(shù)據(jù)量很大,而用戶又不愿意左右滑動(dòng)瀏覽,那么這時(shí)候ViewPager中本來(lái)充滿善意的預(yù)加載就有點(diǎn)令人不爽了。我們能做的就是屏蔽掉ViewPager的預(yù)加載機(jī)制。雖然ViewPager中提供的有setOffscreenPageLimit()來(lái)控制其預(yù)加載的數(shù)目,但是當(dāng)設(shè)置為0后我們發(fā)現(xiàn)其根本沒(méi)效果,這個(gè)的最小值就是1,也就是你只能最少前后各預(yù)加載一頁(yè)。那么,這時(shí)候就得另覓方法了。
以下三種方法是我在學(xué)習(xí)和項(xiàng)目中嘗試過(guò)的,需求實(shí)現(xiàn)了,但各有千秋,可結(jié)合不同場(chǎng)景使用。因?yàn)榇蛩懵B(yǎng)成寫(xiě)博客的習(xí)慣,就總結(jié)在此,也希望對(duì)他人有所借鑒。
方法一
在Fragment可見(jiàn)時(shí)請(qǐng)求數(shù)據(jù)。此方案仍預(yù)加載了前后的頁(yè)面,但是沒(méi)有請(qǐng)求數(shù)據(jù),只有進(jìn)入到當(dāng)前Framgent時(shí)才請(qǐng)求數(shù)據(jù)。
優(yōu)點(diǎn):實(shí)現(xiàn)了數(shù)據(jù)的懶加載
缺點(diǎn):一次仍是三個(gè)Framgment對(duì)象,不是完全意義的懶加載
public class FragmentSample extends Fragment{
...
@Override
public void setUserVisibleHint(boolean isVisibleToUser) {
super.setUserVisibleHint(isVisibleToUser);
if (isVisibleToUser) {
requestData(); // 在此請(qǐng)求數(shù)據(jù)
}
}
...
}
方法二
直接修改ViewPager源碼。通過(guò)查看ViewPager源碼可知,控制其預(yù)加載的是一個(gè)常量DEFAULT_OFFSCREEN_PAGES,其默認(rèn)值為1,表示當(dāng)前頁(yè)面前后各預(yù)加載一個(gè)頁(yè)面,在這里我們直接將其設(shè)置為0即可,即去掉預(yù)加載。但是,這樣有一個(gè)問(wèn)題,那就是在使用其他控件時(shí)需要傳入ViewPager時(shí),這個(gè)就不能用了。
優(yōu)點(diǎn):完全屏蔽掉了預(yù)加載
缺點(diǎn):應(yīng)用太受限制,比如使用ViewPagerIndicator時(shí)需要傳入ViewPager對(duì)象,這時(shí)傻眼了。
// 注意,這是直接拷貝的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;
// 默認(rèn)為1,即前后各預(yù)加載一個(gè)頁(yè)面,設(shè)置為0去掉預(yù)加載
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,結(jié)合PagerAdapter實(shí)現(xiàn)懶加載。該方案是我用到的最完善的方法,完全的懶加載,每次只會(huì)建立一個(gè)Fragment對(duì)象。
優(yōu)點(diǎn):完全屏蔽預(yù)加載
缺點(diǎn):稍微復(fù)雜,但是人家已經(jīng)造好的輪子,直接用吧,很簡(jiǎn)潔
這個(gè)庫(kù)就4個(gè)類,作者通過(guò)繼承ViewPager(保證其普適性)、自定義ViewPagerAdapter和 LazyFragmentPagerAdapter以及設(shè)置懶加載的標(biāo)記接口,很好的實(shí)現(xiàn)了懶加載。感謝作者。
在此貼出關(guān)鍵代碼,有興趣的同學(xué)可以學(xué)習(xí)下。
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一定要實(shí)現(xiàn)接口LazyFragmentPagerAdapter.Laziable。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Android之Viewpager+Fragment實(shí)現(xiàn)懶加載示例
- Android ViewPager動(dòng)態(tài)加載問(wèn)題
- 詳解Android_性能優(yōu)化之ViewPager加載成百上千高清大圖oom解決方案
- android 解決ViewPager加載大量圖片內(nèi)存溢出問(wèn)題
- Android ViewPager制作新手導(dǎo)航頁(yè)(動(dòng)態(tài)加載)
- Android 使用ViewPager自動(dòng)滾動(dòng)循環(huán)輪播效果
- Android ViewPager實(shí)現(xiàn)圖片輪播效果
- Android使用ViewPager實(shí)現(xiàn)自動(dòng)輪播
- Android中用RxJava和ViewPager實(shí)現(xiàn)輪播圖
- Android使用ViewPager加載圖片和輪播視頻
相關(guān)文章
Flutter 實(shí)現(xiàn)虎牙/斗魚(yú) 彈幕功能
這篇文章主要介紹了Flutter 實(shí)現(xiàn)虎牙/斗魚(yú) 彈幕功能,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-04-04
Android基礎(chǔ)總結(jié)篇之三:Activity的task相關(guān)介紹
這篇文章主要介紹了Android基礎(chǔ)總結(jié)篇之三:Activity的task相關(guān)介紹,具有一定的參考價(jià)值,有需要的可以了解一下。2016-11-11
Android Studio 3.6安裝全過(guò)程及AVD安裝運(yùn)行步驟詳解
這篇文章主要介紹了Android Studio 3.6安裝全過(guò)程及AVD安裝運(yùn)行步驟詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-03-03
android輕松管理安卓應(yīng)用中的log日志 發(fā)布應(yīng)用時(shí)log日志全部去掉的方法
android合理的管理log日志,在開(kāi)發(fā)的時(shí)候打印出來(lái),在發(fā)布的時(shí)候,把所有的log日志全部關(guān)掉,下面就把方法給你一一道來(lái)2013-11-11
Android 自定義View結(jié)合自定義TabLayout實(shí)現(xiàn)頂部標(biāo)簽滑動(dòng)效果
小編最近在做app的項(xiàng)目,需要用到tablayout實(shí)現(xiàn)頂部的滑動(dòng)效果,文中代碼用到了自定義item,代碼也很簡(jiǎn)單,感興趣的朋友跟隨腳本之家小編一起看看吧2018-07-07
android計(jì)算器實(shí)現(xiàn)兩位數(shù)的加減乘除
這篇文章主要為大家詳細(xì)介紹了android計(jì)算器實(shí)現(xiàn)兩位數(shù)的加減乘除,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-03-03
Android自定義view實(shí)現(xiàn)車(chē)載可調(diào)整軌跡線
這篇文章主要為大家詳細(xì)介紹了Android自定義view實(shí)現(xiàn)車(chē)載可調(diào)整軌跡線,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-06-06
Android 自定義View手寫(xiě)簽名并保存圖片功能
這篇文章主要介紹了Android 自定義View手寫(xiě)簽名并保存圖片功能,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值 ,需要的朋友可以參考下2020-03-03

