摘要:在,,中調用在,,,,,具體的回收操作,在類的方法中,其中這兩個方法,,方法利用獲取與之綁定的方法回收有兩種情況當滿了,刪除中存放的一個,把這個放入到中放入中,的增和刪都在方法中查刪在方法中的刪查在方法只在其他沒有的方法中調用
dispatchLayout在onLayout,
consumePendingUpdateOperations,
resumeRequestLayout中調用
resumeRequestLayout 在focusSearch,onMeasure,
dispatchLayoutStep1,dispatchLayoutStep2,dispatchLayoutStep3,
ViewFlinger.run
removeAnimatingView
scrollByInternal<-- scrollBy
void dispatchLayout() { if (mState.mLayoutStep == State.STEP_START) { dispatchLayoutStep1(); mLayout.setExactMeasureSpecsFrom(this); dispatchLayoutStep2(); } else if (mAdapterHelper.hasUpdates() || mLayout.getWidth() != getWidth() || mLayout.getHeight() != getHeight()) { // First 2 steps are done in onMeasure but looks like we have to run again due to // changed size. mLayout.setExactMeasureSpecsFrom(this); dispatchLayoutStep2(); } else { // always make sure we sync them (to ensure mode is exact) mLayout.setExactMeasureSpecsFrom(this); } dispatchLayoutStep3(); }
private void dispatchLayoutStep3() { ... ... mViewInfoStore.process(mViewInfoProcessCallback); ... ... }
private final ViewInfoStore.ProcessCallback mViewInfoProcessCallback = new ViewInfoStore.ProcessCallback() { ... ... @Override public void unused(ViewHolder viewHolder) { mLayout.removeAndRecycleView(viewHolder.itemView, mRecycler); } }
RecyclerView.LayoutManager:
public void removeAndRecycleView(View child, Recycler recycler) { removeView(child); recycler.recycleView(child); }
具體的回收操作,在RecyclerView.Recycler類的recycleView方法中,
RecyclerView.Recycler:
public void recycleView(View view) { // This public recycle method tries to make view recycle-able since layout manager // intended to recycle this view (e.g. even if it is in scrap or change cache) ViewHolder holder = getChildViewHolderInt(view); if (holder.isTmpDetached()) { removeDetachedView(view, false); } if (holder.isScrap()) { holder.unScrap(); } else if (holder.wasReturnedFromScrap()){ holder.clearReturnedFromScrapFlag(); } recycleViewHolderInternal(holder); } 其中這兩個方法,getChildViewHolderInt,recycleViewHolderInternal getChildViewHolderInt方法利用View獲取與之綁定的ViewHolder static ViewHolder getChildViewHolderInt(View child) { if (child == null) { return null; } return ((LayoutParams) child.getLayoutParams()).mViewHolder; } recycleViewHolderInternal方法回收ViewHolder,有兩種情況: 1.當mCachedViews滿了,刪除mCachedViews中存放的一個ViewHolder,把這個ViewHoler放入到RecyclerPool中; 2.放入mCachedViews中 void recycleViewHolderInternal(ViewHolder holder) { ... ... if (forceRecycle || holder.isRecyclable()) { if (!holder.hasAnyOfTheFlags(ViewHolder.FLAG_INVALID | ViewHolder.FLAG_REMOVED | ViewHolder.FLAG_UPDATE)) { // Retire oldest cached view final int cachedViewSize = mCachedViews.size(); if (cachedViewSize == mViewCacheMax && cachedViewSize > 0) { recycleCachedViewAt(0); } if (cachedViewSize < mViewCacheMax) { mCachedViews.add(holder); cached = true; } } if (!cached) { addViewHolderToRecycledViewPool(holder); recycled = true; } } ... ... } void recycleCachedViewAt(int cachedViewIndex) { if (DEBUG) { Log.d(TAG, "Recycling cached view at index " + cachedViewIndex); } ViewHolder viewHolder = mCachedViews.get(cachedViewIndex); if (DEBUG) { Log.d(TAG, "CachedViewHolder to be recycled: " + viewHolder); } addViewHolderToRecycledViewPool(viewHolder); mCachedViews.remove(cachedViewIndex); } void addViewHolderToRecycledViewPool(ViewHolder holder) { ViewCompat.setAccessibilityDelegate(holder.itemView, null); dispatchViewRecycled(holder); holder.mOwnerRecyclerView = null; getRecycledViewPool().putRecycledView(holder); }
RecyclerView.RecyclerPool:
public void putRecycledView(ViewHolder scrap) { final int viewType = scrap.getItemViewType(); final ArrayList scrapHeap = getScrapHeapForType(viewType); if (mMaxScrap.get(viewType) <= scrapHeap.size()) { return; } if (DEBUG && scrapHeap.contains(scrap)) { throw new IllegalArgumentException("this scrap item already exists"); } scrap.resetInternal(); scrapHeap.add(scrap); }
mCachedViews, RecyclerPool
mCachedViews的增和刪都在recycleViewHolderInternal方法中,查刪在getScrapViewForPosition方法中
ViewHolder getScrapViewForPosition(int position, int type, boolean dryRun) { ... ... final int cacheSize = mCachedViews.size(); for (int i = 0; i < cacheSize; i++) { final ViewHolder holder = mCachedViews.get(i); // invalid view holders may be in cache if adapter has stable ids as they can be // retrieved via getScrapViewForId if (!holder.isInvalid() && holder.getLayoutPosition() == position) { if (!dryRun) { mCachedViews.remove(i); } if (DEBUG) { Log.d(TAG, "getScrapViewForPosition(" + position + ", " + type + ") found match in cache: " + holder); } return holder; } } ... ... }
RecyclerPool的刪查在
Recycler:
View getViewForPosition(int position, boolean dryRun) { ... ... // 1) Find from scrap by position if (holder == null) { holder = getScrapViewForPosition(position, INVALID_TYPE, dryRun); if (holder != null) { if (!validateViewHolderForOffsetPosition(holder)) { // recycle this scrap if (!dryRun) { // we would like to recycle this but need to make sure it is not used by // animation logic etc. holder.addFlags(ViewHolder.FLAG_INVALID); if (holder.isScrap()) { removeDetachedView(holder.itemView, false); holder.unScrap(); } else if (holder.wasReturnedFromScrap()) { holder.clearReturnedFromScrapFlag(); } recycleViewHolderInternal(holder); } holder = null; } else { fromScrap = true; } } } ... ... if (holder == null) { // fallback to recycler // try recycler. // Head to the shared pool. if (DEBUG) { Log.d(TAG, "getViewForPosition(" + position + ") fetching from shared " + "pool"); } holder = getRecycledViewPool().getRecycledView(type); if (holder != null) { holder.resetInternal(); if (FORCE_INVALIDATE_DISPLAY_LIST) { invalidateDisplayListInt(holder); } } } ... ... }
getViewForPosition方法只在LinearLayoutManager.LayoutState.next,其他LayoutManager沒有的方法中調用
View next(RecyclerView.Recycler recycler) { if (mScrapList != null) { return nextViewFromScrapList(); } final View view = recycler.getViewForPosition(mCurrentPosition); mCurrentPosition += mItemDirection; return view; }
LinearLayoutManager.java:
void layoutChunk(RecyclerView.Recycler recycler, RecyclerView.State state, LayoutState layoutState, LayoutChunkResult result) { View view = layoutState.next(recycler); ... ... LayoutParams params = (LayoutParams) view.getLayoutParams(); if (layoutState.mScrapList == null) { if (mShouldReverseLayout == (layoutState.mLayoutDirection == LayoutState.LAYOUT_START)) { addView(view); } else { addView(view, 0); } } else { if (mShouldReverseLayout == (layoutState.mLayoutDirection == LayoutState.LAYOUT_START)) { addDisappearingView(view); } else { addDisappearingView(view, 0); } } measureChildWithMargins(view, 0, 0); ... ... layoutDecoratedWithMargins(view, left, top, right, bottom); ... ... } int fill(RecyclerView.Recycler recycler, LayoutState layoutState, RecyclerView.State state, boolean stopOnFocusable) { ... ... while ((layoutState.mInfinite || remainingSpace > 0) && layoutState.hasMore(state)) { layoutChunkResult.resetInternal(); layoutChunk(recycler, state, layoutState, layoutChunkResult); if (layoutChunkResult.mFinished) { break; } ... ... } ... ... } public void onLayoutChildren(RecyclerView.Recycler recycler, RecyclerView.State state) { // layout algorithm: // 1) by checking children and other variables, find an anchor coordinate and an anchor // item position. // 2) fill towards start, stacking from bottom // 3) fill towards end, stacking from top // 4) scroll to fulfill requirements like stack from bottom. ... ... }
RecyclerView.java:
private void dispatchLayoutStep1() { ... ... mLayout.onLayoutChildren(mRecycler, mState); ... ... }
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/70765.html
摘要:其實通過父類的這個方法之后會調用它的方法,這個名字熟悉自定義的童鞋都知道了。 為什么要寫這篇源碼解析呢? 我一直在說RecyclerView是一個值得深入學習,甚至可以說是一門具有藝術性的控件。那到底哪里值得我們花時間去深入學習呢。沒錯了,就是源碼的設計。但是看源碼其實是一件不簡單的事情,就拿RecyclerView的源碼來說,打開源碼一看,往下拉啊拉啊,我擦,怎么還沒到頭,汗.......
摘要:缺點自動裝箱的存在意味著每一次插入都會有額外的對象創建。對象本身是一層額外需要被創建以及被垃圾回收的對象。相較于我們舍棄了和類型的放棄了并依賴于二分法查找。 目錄介紹 25.0.0.0 請說一下RecyclerView?adapter的作用是什么,幾個方法是做什么用的?如何理解adapter訂閱者模式? 25.0.0.1 ViewHolder的作用是什么?如何理解ViewHolder...
閱讀 2617·2021-11-16 11:40
閱讀 3412·2021-11-08 13:26
閱讀 877·2021-10-28 09:32
閱讀 3535·2021-09-13 10:26
閱讀 809·2019-08-30 15:55
閱讀 783·2019-08-30 15:44
閱讀 1912·2019-08-30 15:44
閱讀 1759·2019-08-30 13:48