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