首页 > 代码库 > LayoutInflater.inflate详解

LayoutInflater.inflate详解

介绍常见inflate方法



在日常开发中经常会用到通过资源id去获取view的场景,我们通常有四种方式去获取view,分别是以下四种:

//1,通过系统服务获取布局加载器
LayoutInflater inflater = (LayoutInflater) getSystemService(Context.LAYOUT_INFLATER_SERVICE);


View view = inflater.inflate(resource,root,attachToRoot);


//2,通过activity中的getLayoutInflater()方法
View view = getLayoutInflater().inflate(resource,root,attachToRoot);


//3,通过View的静态inflate()方法
View view = View.inflate(resource,root,attachToRoot);


//4,通过LayoutInflater的inflate()方法
View view = LayoutInflater.from(this).inflate(resource,root,attachToRoot);


通过对上述方法的源码的分析,很容易看出来这些方法都是最终调用了方式1,获取系统布局加载器的方式,来进行获取`View`。


这里我列举的并没有`inflate(int resource, ViewGroup root)`这个重载方法,是因为他们最终都会到调用为`inflate(int resource, ViewGroup root, boolean attachToRoot)`方法,如下:

 public View inflate(int resource, ViewGroup root) {
        return inflate(resource, root, root != null);
 }
 
 public View inflate(int resource, ViewGroup root, boolean attachToRoot) {
        if (DEBUG) System.out.println("INFLATING from resource: " + resource);
        XmlResourceParser parser = getContext().getResources().getLayout(resource);
        try {
            return inflate(parser, root, attachToRoot);
        } finally {
            parser.close();
        }
 }

这里要说明一下,其实最终要调用的是`inflate(XmlPullParser parser, ViewGroup root, boolean attachToRoot)`方法,只不过这里把布局资源解析为了XmlPull解析器,这里就不针对`XmlPullParser`进行研究了。


针对传入的参数不同进行分析

通过对`inflate(XmlPullParser parser, ViewGroup root, boolean attachToRoot)`方法的源码进行解读,`XmlPullParser parser`是对传入`int resource`的xml解析器,不用去主要考虑,那就需要考虑传入`ViewGroup root, boolean attachToRoot`的值不同,会出现什么结果呢?


源码中有几个重要的代码块:

if (root != null) {
if (DEBUG) {
System.out.println("Creating params from root: " +
                                    root);
}
// 如果root不等于null,获取它的LayoutParams
  params = root.generateLayoutParams(attrs);
if (!attachToRoot) {
//attachToRoot等于false,把root的LayoutParams属性给temp
temp.setLayoutParams(params);
}
}

//attachToRoot等于true,将temp加入到root这个viewGroup中
if (root != null && attachToRoot) {
root.addView(temp, params);
}


// root等于null,attachToRoot等于false,直接把temp赋值给返回结果
if (root == null || !attachToRoot) {
  result = temp;
}

根据不同的传值进行实现



两个布局文件,一个作为root,一个作为我们的要获取的view


activity_my.xml
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MyActivity">


    <TextView
        android:text="@string/hello_world"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content" />


</RelativeLayout>

view.xml
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="120dp"
    android:layout_height="120dp"
    android:background="@color/blue">


    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="hello inflate"
        android:textColor="@android:color/white" />


</LinearLayout>




1,rootView等于null,attachToRoot等于false

protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
    view = getLayoutInflater().inflate(R.layout.view, null, false);
    setContentView(view);
}



本身root为null,就不会去获取view的LayoutParams,直接走`result = temp;`返回result,此时result等于view


为什么会充满屏幕呢,因为当把view设置到activity的视图时,系统会取当前window的LayoutParm作为view的LayoutParm




2,rootView等于null,attachToRoot等于true

protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
    view = getLayoutInflater().inflate(R.layout.view, null, true);
    setContentView(view);
}


root等于null,依然走`result = temp;`,此时result等于view


为什么会充满屏幕呢,因为当把view设置到activity的视图时,系统会取当前window的LayoutParm作为view的LayoutParm


3,rootView不等于null,attachToRoot等于false

protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
rootView = getLayoutInflater().inflate(R.layout.activity_my, null);
    view = getLayoutInflater().inflate(R.layout.view,(ViewGroup)rootView,false);
    setContentView(view);
}


root不等于null,attachToRoot等于false,会走取view的LayoutParams并且赋值给temp,再走`result = temp;`,此时result等于view


为什么会充满屏幕呢,因为当把view设置到activity的视图时,系统会取当前window的LayoutParm作为view的LayoutParm


4,rootView不等于null,attachToRoot等于true
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
rootView = getLayoutInflater().inflate(R.layout.activity_my, null);
    view = getLayoutInflater().inflate(R.layout.view,(ViewGroup)rootView,true);
    setContentView(view);
}

rootView不等于null,attachToRoot等于true,会首先获取view的Params,再走`root.addView(temp, params)`,也就是,把view先放入的root这个ViewGroup中,再返回result,因为初始化的时候result就等于root,此时返回的就是包含有子View的root。


这里因为root的布局为RelativeLayout,我们把view加入到root中,view本身保留了自有的LayoutParm


最后我要吐槽,为什么csdn不支持markdown,我都是先用Mou写好的,贴过来竟然要自己重新排版。。。

LayoutInflater.inflate详解