Android findViewById源码解析

从页面的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过程:

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值