在阅读源代码时候的一点小发现
activity的视图加载在大家熟悉的onCreate(int layoutResIDent)重载方法中调用,先是activity类中的setContentView(int layoutResID)调用Window类中的setContentView(int layoutResID),如下:
public void setContentView(int layoutResID){
getWindow().setContentView(layoutResID);
initWindowDecorActionBar();
}
但是在Window类中的setContentView(int layoutResID)是个抽象方法,即Window类也是个抽象类,由PhoneWindow类继承Window类,并实现了Window类中的setContentView(int layoutResID)方法,如下:
public void setContentView(int layoutResID){
if(mContentParent == null){
installDecor();
}else{
cContentParent.removeAllView();
}
mLayoutInflater.inflate(layoutResID,mContentParent);
final Callback cb = getCallback();
if(cb != null && isDestroyed()){
cb.onContentChanged();
}
}
判断mContentParent是否为null,如果是执行installDecor(),如果不是则移除mContentParent的所有View,总而言之,就是要在这里放一个干净的mContentParent,之后的代码就是用LayoutInflater将xml布局加载到mContentParent,然后通过回调告诉Activity。
类图如下:
同理关于findViewById(int id)也是先从activity中加载window,然后通过window加载view,如下:
在activity中的定义:
public View findViewById(int id){
return getWindow().findViewById(id);
}
在Window中的定义:
<pre name="code" class="java">public View findViewById(int id){
return getDecorView(),findViewById(id);
}
在View中的定义:
public final View findViewById(int id){
if(id < 0){
return null;
}
return findViewTraversal(id);
}
protected View findViewTraversal(int id){
if(id == mID){
return this;
}
return null;
}