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)文章
Android源碼導(dǎo)入AndroidStudio或IntelliJ?IDEA的方法
這篇文章主要介紹了Android源碼導(dǎo)入AndroidStudio或IntelliJ?IDEA的方法,用idegen來(lái)生成針對(duì)AndroidStudio或IntelliJ?IDEA的Android系統(tǒng)源代碼工程配置文件,需要的朋友可以參考下2022-08-08andriod開(kāi)發(fā)之Activity的渲染機(jī)制
本文給大家分享的是在andriod開(kāi)發(fā)中經(jīng)常需要用到的Activity的渲染機(jī)制的詳細(xì)說(shuō)明,主要是通過(guò)實(shí)例給大家講解Activity是如何畫(huà)到屏幕上的,希望大家能夠喜歡2018-03-03Android應(yīng)用開(kāi)發(fā)中Fragment存儲(chǔ)功能的基本用法
這篇文章主要介紹了Android應(yīng)用開(kāi)發(fā)中使用Fragment存儲(chǔ)功能的基本用法,包括對(duì)Fragment的非中斷保存setRetaineInstance的講解,需要的朋友可以參考下2016-02-02Android實(shí)用控件自定義逼真相機(jī)光圈View
這篇文章主要為大家詳細(xì)介紹了Android實(shí)用控件自定義逼真相機(jī)光圈,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-08-08Android Studio添加第三方庫(kù)的注意事項(xiàng)
這篇文章給大家介紹的是Android Studio添加第三方庫(kù)遇到的一些坑,以及對(duì)應(yīng)的解決辦法,有需要的可以參考借鑒。2016-09-09Android中判斷網(wǎng)絡(luò)是否連接實(shí)例詳解
這篇文章主要介紹了Android中判斷網(wǎng)絡(luò)是否連接實(shí)例詳解的相關(guān)資料,需要的朋友可以參考下2017-01-01Android簡(jiǎn)易圖片瀏覽器的實(shí)現(xiàn)
最近做了一個(gè)圖片瀏覽小程序,本文主要介紹了Android簡(jiǎn)易圖片瀏覽器的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-03-03Android3.0 ActionBar導(dǎo)航標(biāo)題欄使用解析
這篇文章主要為大家詳細(xì)解析了Android3.0 ActionBar導(dǎo)航標(biāo)題欄的使用方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-01-01