欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

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

 更新時(shí)間:2017年03月06日 15:02:48   作者:AnswerZhao_  
這篇文章主要介紹了android實(shí)現(xiàn)ViewPager懶加載的三種方法,懶加載在項(xiàng)目運(yùn)用中很廣泛,可以提高運(yùn)行速度,有興趣的可以了解一下。

在項(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)潔

代碼下載:LazyViewPager_jb51.rar

這個(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í)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

最新評(píng)論