溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

android實(shí)現(xiàn)ViewPager懶加載的三種方法

發(fā)布時間:2020-10-07 17:57:47 來源:腳本之家 閱讀:423 作者:AnswerZhao_ 欄目:移動開發(fā)

在項(xiàng)目中ViewPager和Fragment接口框架已經(jīng)是處處可見,但是在使用中,我們肯定不希望用戶在當(dāng)前頁面時就在前后頁面的數(shù)據(jù),加入數(shù)據(jù)量很大,而用戶又不愿意左右滑動瀏覽,那么這時候ViewPager中本來充滿善意的預(yù)加載就有點(diǎn)令人不爽了。我們能做的就是屏蔽掉ViewPager的預(yù)加載機(jī)制。雖然ViewPager中提供的有setOffscreenPageLimit()來控制其預(yù)加載的數(shù)目,但是當(dāng)設(shè)置為0后我們發(fā)現(xiàn)其根本沒效果,這個的最小值就是1,也就是你只能最少前后各預(yù)加載一頁。那么,這時候就得另覓方法了。

以下三種方法是我在學(xué)習(xí)和項(xiàng)目中嘗試過的,需求實(shí)現(xiàn)了,但各有千秋,可結(jié)合不同場景使用。因?yàn)榇蛩懵B(yǎng)成寫博客的習(xí)慣,就總結(jié)在此,也希望對他人有所借鑒。

方法一

在Fragment可見時請求數(shù)據(jù)。此方案仍預(yù)加載了前后的頁面,但是沒有請求數(shù)據(jù),只有進(jìn)入到當(dāng)前Framgent時才請求數(shù)據(jù)。

優(yōu)點(diǎn):實(shí)現(xiàn)了數(shù)據(jù)的懶加載

缺點(diǎn):一次仍是三個Framgment對象,不是完全意義的懶加載

public class FragmentSample extends Fragment{
  ... 
  @Override
  public void setUserVisibleHint(boolean isVisibleToUser) {
    super.setUserVisibleHint(isVisibleToUser);
    if (isVisibleToUser) {
      requestData(); // 在此請求數(shù)據(jù)
    }
  }
  ...
}

方法二

直接修改ViewPager源碼。通過查看ViewPager源碼可知,控制其預(yù)加載的是一個常量DEFAULT_OFFSCREEN_PAGES,其默認(rèn)值為1,表示當(dāng)前頁面前后各預(yù)加載一個頁面,在這里我們直接將其設(shè)置為0即可,即去掉預(yù)加載。但是,這樣有一個問題,那就是在使用其他控件時需要傳入ViewPager時,這個就不能用了。

優(yōu)點(diǎn):完全屏蔽掉了預(yù)加載

缺點(diǎn):應(yīng)用太受限制,比如使用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;
   // 默認(rèn)為1,即前后各預(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)懶加載。該方案是我用到的最完善的方法,完全的懶加載,每次只會建立一個Fragment對象。

優(yōu)點(diǎn):完全屏蔽預(yù)加載

缺點(diǎn):稍微復(fù)雜,但是人家已經(jīng)造好的輪子,直接用吧,很簡潔

代碼下載:LazyViewPager_jb51.rar

這個庫就4個類,作者通過繼承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。

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持億速云。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI