ViewPaper 系列 — 子View的缓存基本原理

很多时候,我们想方设法找自定义控件的时候,有没有想过,android本身的自定义控件就已经很是丰富多彩了。

简介

viewpager,一个我们经常用到的控件,让我们一起简单看看其中的实现以及整个过程的想法

好奇的驱使

一直都很好奇,viewpager为什么能够滑动呢?其中,google工程师们是怎么让其实现的呢?怀揣着这样的心理我们废话不多说,一起来看一看吧(由于水平有限,也只是简单了解一下-

探求过程

第一点

如下,viewpager继承至viewgroup,既然是viewgroup,那它应该就是在onMeasure()中,onLayout()中进行子view的排版的咯。疑问出现,那它是怎么添加子view的呢,子view是怎么超出一个屏幕的呢?怀着疑问,这时就准备看onMeasure()方法了。

public class ViewPager extends ViewGroup{
        ......
}
第二点

接下来,我们来到onMeasure

code1.png

我们稍微翻译一下(有错误,望指出):为了简单的实现,内部的size就暂时先设置为0,在添加或者删除控件之前,我们是不知道其大小的,并且我们也不想这个时候改变大小引起layout的调用。

接下来的代码

final int measuredWidth = getMeasuredWidth();
final int maxGutterSize = measuredWidth / 10;
mGutterSize = Math.min(maxGutterSize, mDefaultGutterSize);
// 由这里可以看出,给出子view的宽高就是viewpaper的宽高减去相应的padding值
int childWidthSize = measuredWidth - getPaddingLeft() - getPaddingRight();
int childHeightSize = getMeasuredHeight() - getPaddingTop() - getPaddingBottom();

int size = getChildCount();

看到这里,是否有一个疑问?child从何而来,回忆在使用viewpager时,初始化后一般都是设置适配器,那么就去看下setAdapter();虽然跳来跳去有点乱,但是,主要是为了描述整个看代码的过程

public void setAdapter(PagerAdapter adapter) {
        ...
        final PagerAdapter oldAdapter = mAdapter;
        mAdapter = adapter;
        mExpectedAdapterCount = 0;

        if (mAdapter != null) {
            if (mObserver == null) {
                mObserver = new PagerObserver();
            }
            mAdapter.setViewPagerObserver(mObserver);
            mPopulatePending = false;
            final boolean wasFirstLayout = mFirstLayout;
            mFirstLayout = true;
            mExpectedAdapterCount = mAdapter.getCount();
            if (mRestoredCurItem >= 0) {
                mAdapter.restoreState(mRestoredAdapterState, mRestoredClassLoader);
                setCurrentItemInternal(mRestoredCurItem, false, true);
                mRestoredCurItem = -1;
                mRestoredAdapterState = null;
                mRestoredClassLoader = null;
            } else if (!wasFirstLayout) {
                populate();
            } else {
                // 第一次调用时,会走这里
                requestLayout();
            }
        }
        ...
    }

也就是说setAdapter()在第一次调用的时候仅仅使其调用了requestLayout(),并未发现add子view。于是,回过头继续看onMeasure()。所以第一次调用onMeasure()时,上方那句代码的size就为0;
int size = getChildCount();

最后就落到了这一段代码上

        mInLayout = true;
        populate();
        mInLayout = false;
        // populate()方法前后,size发生了变化,于是这个方法就成了add子view的关键方法
        size = getChildCount();
        Log.d(TAG_TEST,"after populate size: "+ size);

        for (int i = 0; i < size; ++i) {
            final View child = getChildAt(i);
            if (child.getVisibility() != GONE) {
                if (DEBUG) Log.v(TAG, "Measuring #" + i + " " + child
                        + ": " + mChildWidthMeasureSpec);

                final LayoutParams lp = (LayoutParams) child.getLayoutParams();
                if (lp == null || !lp.isDecor) {
                    final int widthSpec = MeasureSpec.makeMeasureSpec(
                            (int) (childWidthSize * lp.widthFactor), MeasureSpec.EXACTLY);
                    child.measure(widthSpec, mChildHeightMeasureSpec);
                }
            }
        }

看到这里,就知道了 populate()是较为重要的一个方法,接下来进入populate()方法

  • 一个小点

以下这个预加载页数是可以通过设置进行改变的,看到这里有一个小小的猜想,我们能不能把这个预加载数量设置的大一点,让那种子条目较多的viewpaper多缓存几个页面呢?(具体有待验证)

        // 这个值应该就是预加载页面的数量值
        final int pageLimit = mOffscreenPageLimit;
        final int startPos = Math.max(0, mCurItem - pageLimit);
  • 找到当前的子条目赋值给curItem

最开始没有子view,这个方法便略过,当mItems有值之后,便会根据当中存储的position来从缓存中查找出当前应该显示在屏幕上的子view

        int curIndex = -1;
        ItemInfo curItem = null;
        for (curIndex = 0; curIndex < mItems.size(); curIndex++) {
            final ItemInfo ii = mItems.get(curIndex);
            if (ii.position >= mCurItem) {
                if (ii.position == mCurItem) curItem = ii;
                break;
            }
        }
  • add子条目

当上面的代码没有找到当前view时,一般情况下是缓存子view的mItems还未加载过这个view或是该view在滑动的过程中被移除了。

        if (curItem == null && N > 0) {
            curItem = addNewItem(mCurItem, curIndex);
        }
  • 遍历当前item左边的所有子view

以下这个方法很有意思,通过预加载数作为判断条件,把需要加载的view都加载了一遍,触发adapter的初始化和销毁item的操作

            for (int pos = mCurItem - 1; pos >= 0; pos--) {
                // 当子view超出了预加载view的范围,那么就会进行移除操作
                // extraWidthLeft 这个值会在下面累加
                // startPos为预加载项的起始位置。
                if (extraWidthLeft >= leftWidthNeeded && pos < startPos) {
                    if (ii == null) {
                        break;
                    }
                    if (pos == ii.position && !ii.scrolling) {
                        mItems.remove(itemIndex);
                        mAdapter.destroyItem(this, pos, ii.object);
                        if (DEBUG) {
                            Log.i(TAG, "populate() - destroyItem() with pos: " + pos +
                                    " view: " + ((View) ii.object));
                        }
                        itemIndex--;
                        curIndex--;
                        ii = itemIndex >= 0 ? mItems.get(itemIndex) : null;
                    }
                // 预加载项默认是1,所以第二个条件成立默认是当前项的左边第一个view
                } else if (ii != null && pos == ii.position) {
                    extraWidthLeft += ii.widthFactor;
                    // 左边移动了一个pos
                    itemIndex--;
                    // 左移后,ii被重新赋值到左移一个pos之后的view,这里比较巧妙
                    // 如果ii为null,则下一次for循环就分两种情况,第一种情况:预加载项为默认,也就是1,则会进入第一个if判断。
                    // 大于1的话,则会进入下面的else判断,继续加载其他的预加载view。 
                    ii = itemIndex >= 0 ? mItems.get(itemIndex) : null;
                } else {
                    // 加载预加载子view
                    ii = addNewItem(pos, itemIndex + 1);
                    extraWidthLeft += ii.widthFactor;
                    curIndex++;
                    ii = itemIndex >= 0 ? mItems.get(itemIndex) : null;
                }
            }

注意 addNewItem这个方法,它其中会去调用adapter的初始化view的方法。

接下来,下面的代码就是加载右边的所有view。与加载左边的view类似,所以就不再累赘。
说到这里view的添加就基本可以告一段落了。从中就了解了viewpager缓存view的基本原理:先找到当前view,然后,从此处开始通过设置的预加载数目为条件,分别向前和向后遍历,从缓存子view的集合中取出相对应的元素,并且重新加载未缓存的元素。(未完,待续)

由于篇幅的原因,接下来的内容下一篇文章(手势移动时,viewpager的一些基本处理过程)接着讲。由于自身水平有限,写的不对的地方还望大家体谅,多多指出。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,657评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,662评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,143评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,732评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,837评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,036评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,126评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,868评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,315评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,641评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,773评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,859评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,584评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,676评论 2 351

推荐阅读更多精彩内容