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)成寫博客的習(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-08
andriod開(kāi)發(fā)之Activity的渲染機(jī)制
本文給大家分享的是在andriod開(kāi)發(fā)中經(jīng)常需要用到的Activity的渲染機(jī)制的詳細(xì)說(shuō)明,主要是通過(guò)實(shí)例給大家講解Activity是如何畫到屏幕上的,希望大家能夠喜歡2018-03-03
Android應(yīng)用開(kāi)發(fā)中Fragment存儲(chǔ)功能的基本用法
這篇文章主要介紹了Android應(yīng)用開(kāi)發(fā)中使用Fragment存儲(chǔ)功能的基本用法,包括對(duì)Fragment的非中斷保存setRetaineInstance的講解,需要的朋友可以參考下2016-02-02
Android實(shí)用控件自定義逼真相機(jī)光圈View
這篇文章主要為大家詳細(xì)介紹了Android實(shí)用控件自定義逼真相機(jī)光圈,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-08-08
Android Studio添加第三方庫(kù)的注意事項(xiàng)
這篇文章給大家介紹的是Android Studio添加第三方庫(kù)遇到的一些坑,以及對(duì)應(yīng)的解決辦法,有需要的可以參考借鑒。2016-09-09
Android中判斷網(wǎng)絡(luò)是否連接實(shí)例詳解
這篇文章主要介紹了Android中判斷網(wǎng)絡(luò)是否連接實(shí)例詳解的相關(guān)資料,需要的朋友可以參考下2017-01-01
Android簡(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-03
Android3.0 ActionBar導(dǎo)航標(biāo)題欄使用解析
這篇文章主要為大家詳細(xì)解析了Android3.0 ActionBar導(dǎo)航標(biāo)題欄的使用方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-01-01

