Android findViewById源码解析

作者: 奔跑吧李博 | 来源:发表于2020-08-20 11:54 被阅读0次

从页面的findViewById入手源码,findview调用的AppCompatActivity中的findViewById方法:

    @Override
    public <T extends View> T findViewById(@IdRes int id) {
        return getDelegate().findViewById(id);
    }

获取了AppCompatActivity中的抽象类AppCompatDelegate,它声明了AppCompatActivity所需的各个抽象方法,它的实现类是AppCompatDelegateImpl,那么是由AppCompatDelegateImpl调用的findViewById。

    @Nullable
    @Override
    public <T extends View> T findViewById(@IdRes int id) {
        ensureSubDecor();
        return (T) mWindow.findViewById(id);
    }

转战到了window中的findViewById

    @Nullable
    public <T extends View> T findViewById(@IdRes int id) {
        return getDecorView().findViewById(id);
    }

getDecorView是window类中的一个抽象方法,返回类型为View。这里不细将DecorView,感兴趣的伙伴可以去深究一下。

于是就是调用了View中的findViewById方法,判断当前view的mId是不是该要匹配的id,是就返回了当前要找的view,否则返回空。

    @Nullable
    public final <T extends View> T findViewById(@IdRes int id) {
        if (id == NO_ID) {
            return null;
        }
        return findViewTraversal(id);
    }

    protected <T extends View> T findViewTraversal(@IdRes int id) {
        if (id == mID) {
            return (T) this;
        }
        return null;
    }

而View中的id是通过xml布局中设置的id,赋值给mID属性,如果不设id,那么也会调用generateViewId按照规则生成一个id。

    public void setId(@IdRes int id) {
        mID = id;
        if (mID == View.NO_ID && mLabelForId != View.NO_ID) {
            mID = generateViewId();
        }
    }

    public static int generateViewId() {
        for (;;) {
            final int result = sNextGeneratedId.get();
            // aapt-generated IDs have the high byte nonzero; clamp to the range under that.
            int newValue = result + 1;
            if (newValue > 0x00FFFFFF) newValue = 1; // Roll over to 1, not 0.
            if (sNextGeneratedId.compareAndSet(result, newValue)) {
                return result;
            }
        }
    }

在实际场景中,contentview的父布局是为ViewGroup类型的,ViewGroup继承自View,ViewGroup中重写了findViewTraversal方法,用于查找view控件,方法代码如下:

    @Override
    protected <T extends View> T findViewTraversal(@IdRes int id) {
        if (id == mID) {
            return (T) this;
        }

        final View[] where = mChildren;
        final int len = mChildrenCount;

        for (int i = 0; i < len; i++) {
            View v = where[i];

            if ((v.mPrivateFlags & PFLAG_IS_ROOT_NAMESPACE) == 0) {
                v = v.findViewById(id);

                if (v != null) {
                    return (T) v;
                }
            }
        }

        return null;
    }

在ViewGroup中的查找某个控件,如果id为自己的mId,那么就是查询到自己并返回。否则就要通过遍历子view,如果子view是View,就调用findViewById判断id,如果是ViewGroup,就递归调用findViewTraversal方法,遍历查找直到查到返回或所有子view查询完成结束。

总结findViewById过程:

相关文章

网友评论

    本文标题:Android findViewById源码解析

    本文链接:https://www.haomeiwen.com/subject/fulsjktx.html