LayoutInflater大家都不陌生吧,它主要用于加载布局的。
1、Activity的setContentView加载布局过程
大家都知道Activity加载布局是通过setContentView来完成的,其实内部也是通过LayoutInflater来加载布局的。我们一起来看下setContentView的内部实现(简书没有代码模块,我复制了部分代码):
Activity中的部分源码:
public class Activity extends ContextThemeWrapper implements LayoutInflater.Factory2, Window.Callback, KeyEvent.Callback, OnCreateContextMenuListener, ComponentCallbacks2, Window.OnWindowDismissedCallback, WindowControllerCallback {
... ...
/**
*Retrieve the current {@linkandroid.view.Window} for the activity.
*This can be used to directly access parts of the Window API that
*are not available through Activity/Screen.
*
*@returnWindow The current window, or null if the activity is not visual.
*/
public Window getWindow() {
return mWindow;
}
... ...
/**
*Set the activity content from a layout resource. The resource will be inflated, adding
*all top-level views to the activity.
*@paramlayoutResIDResource ID to be inflated.
*@see#setContentView(android.view.View)
*@see#setContentView(android.view.View, android.view.ViewGroup.LayoutParams)
*/
public void setContentView(@LayoutResintlayoutResID) {
getWindow().setContentView(layoutResID);
initWindowDecorActionBar();
}
... ...
final voidattach(Context context,ActivityThread aThread, Instrumentation instr,IBinder token, intident, Application application,Intent intent,ActivityInfo info,CharSequence title,Activity parent,String id,NonConfigurationInstances lastNonConfigurationInstances,Configuration config,String referrer,IVoiceInteractor voiceInteractor,Window window) {
attachBaseContext(context);
mFragments.attachHost(null);
mWindow=new PhoneWindow(this,window);
}
}
由上述Activity的源码可以看到,setContentView是调用PhoneWindow的setContentView方法。
我们继续看PhoneWindow的源码:
public class PhoneWindow extends Window implements MenuBuilder.Callback {
... ...
@Override
public void setContentView(int layoutResID) {
if (mContentParent == null) {
installDecor();
} else {
mContentParent.removeAllViews();
}
mLayoutInflater.inflate(layoutResID, mContentParent);
final Callback cb = getCallback();
if (cb != null && !isDestroyed()) {
cb.onContentChanged();
}
}
......
}
源码PhoneWindow的setContentView最终也是用LayoutInflater来加载布局的。所以Activity的setContentView是通过LayoutInflater来加载布局的。
2.获得 LayoutInflater 实例的三种方式:
1.LayoutInflater inflater = getLayoutInflater(); //调用Activity的getLayoutInflater()
2.LayoutInflater localinflater =(LayoutInflater)context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
3.LayoutInflater inflater = LayoutInflater.from(context);
这三种方式本质上是相同的,1和3最后还是通过2获得的。
首先看下源码Activity中getLayoutInflater()的方法:
/**
* Convenience for calling
* {@linkandroid.view.Window#getLayoutInflater}.
*/
@NonNull
public LayoutInflater getLayoutInflater() {
return getWindow().getLayoutInflater();
}
Activity中LayoutInflater 也是通过PhoneWindow的getLayoutInflater()方法,我们继续跟踪
源码Activity中getLayoutInflater()的方法:
public PhoneWindow(Context context) {
super(context);
mLayoutInflater = LayoutInflater.from(context);
}
... ...
/**
* Return a LayoutInflater instance that can be used to inflate XML view layout
*
* @return LayoutInflater The shared LayoutInflater.
*/
@Override
public LayoutInflater getLayoutInflater() {
return mLayoutInflater;
}
从PhoneWindow的源码中可以看出 方法1的LayoutInflater是通过方法3得到的。
我们继续看下源码LayoutInflater中from()方法:
/**
* Obtains the LayoutInflater from the given context.
*/
public static LayoutInflater from(Context context) {
LayoutInflater LayoutInflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
if(LayoutInflater ==null) {
throw newAssertionError("LayoutInflater not found.");
}
returnLayoutInflater;
}
LayoutInflater类的from()方法最终是通过方式2获取。
方式1是通过方式3获取的,方式3最终是通过方式2获取的.方式1、2、3本质是相同的。
3.关于LayoutInflater类inflate(intresource, ViewGroup root, boolean attachToRoot)方法三个参数的含义
转载:关于LayoutInflater类inflate(intresource, ViewGroup root, boolean attachToRoot)方法三个参数的含义
resource:需要加载布局文件的id,意思是需要将这个布局文件中加载到Activity中来操作。
root:需要附加到resource资源文件的根控件,什么意思呢,就是inflate()会返回一个View对象,如果第三个参数attachToRoot为true,就将这个root作为根对象返回,否则仅仅将这个root对象的LayoutParams属性附加到resource对象的根布局对象上,也就是将布局文件resource的布局参数转换为外层root可以接受的类型,比如root是一个LinearLayout自己要转换的resource里面有layout_width=”fill_parent”,和layout_height=”fill_parent”参数,但是这些参数没有外部环境,它们对应的对象都是ViewGroup.LayoutParams对象,root参数让系统将ViewGroup.LayoutParams对象转换为LinearLayout.LayoutParams对象。
attachToRoot:是否将root附加到布局文件的根视图上