android - LayoutInflater源码解析

LayoutInflater

Instantiates a layout XML file into its corresponding {@link android.view.View} objects. It is never used directly. Instead, use {@link android.app.Activity#getLayoutInflater()} or {@link Context#getSystemService} to retrieve a standard LayoutInflater instance

LayoutInflater用于实例化一个XML 布局文件成相应的View。

它不能直接获取通过 new 的方式获得,可以通过以下三种获取方式:

  1. LayoutInflater LayoutInflater =(LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
  2. LayoutInflater layoutInflater = LayoutInflater.from(context);
  3. LayoutInflater layoutInflater = Activity.getLayoutInflater();

第二种方式是第一种的缩写,第三种获取的是Activity私有成员属性mWindow的LayoutInflater(在我们setContentView前,Activity就需要去加载DecorView)。

/**
     * 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 new AssertionError("LayoutInflater not found.");
        }
        return LayoutInflater;
    }

inflate overload

在拿到实例后,为了实现其加载XML文件成View的职能,我们需要调用 public View inflate()
方法,该方法有多个重载。如下:

1. public View inflate(XmlPullParser parser, @Nullable ViewGroup root, boolean attachToRoot)
2. public View inflate(@LayoutRes int resource, @Nullable ViewGroup root, boolean attachToRoot)
3. public View inflate(@LayoutRes int resource, @Nullable ViewGroup root)
4. public View inflate(XmlPullParser parser, @Nullable ViewGroup root)

这些重载方法其实最终调用的都是 第一个
我们来看它是怎么汇流到第一个方法。

public View inflate(@LayoutRes int resource, @Nullable ViewGroup root, boolean attachToRoot) {
    final XmlResourceParser parser = getContext().getResources().getLayout(resource);
    return inflate(parser, root, attachToRoot);
}

传入的R.layout.main_activity XML布局文件,会被转换成XmlResourceParser 接口类型,然后调用第一个方法。
那不传入attachToRoot参数呢?

    public View inflate(XmlPullParser parser, @Nullable ViewGroup root) {
        return inflate(parser, root, root != null);
    }

@Nullable 注解表示,这个参数可以传空。

  • 当root为空时,调用第一个方法,并传入inflate(parser, root, false)
  • root不为空时,调用第一个方法,并传入inflate(parser, root, true)

inflate @param

终于到达 public View inflate(XmlPullParser parser, @Nullable ViewGroup root, boolean attachToRoot) 方法了。
我们来看它的参数设置

  • @param resource Xml布局文件的ID (如:R.layout.main_activity)
  • @param root 生成View的父视图,如果不为空,就套LayoutParams到View上。
  • @param attachToRoot 是否将生成的View添加到root上(也就是root.addView(view),当attachToRoot为true时,将生成的View添加到root上。并且返回的是root
  • @return
    • 当attachToRoot为true,返回已经添加View的root。
    • 当attachToRoot为false,返回View。

其实还是很清晰的,我们捋一捋,resource参数就是布局xml文件,root决定LayoutParamsattachToRoot决定返回值。

研究完它的使用,我们继续看它到底是怎么样将一个xml文件加载成java里的View对象。

inflate function

    final Context inflaterContext = mContext;
    View result = root;
    int type;
    final String name = parser.getName();
    
    //如果开头为<merge>标签则需要传入正确的root 和attachToRoot
    if (TAG_MERGE.equals(name)) {
        if (root == null || !attachToRoot) {
            throw new InflateException("<merge /> can be used only with a valid "
                    + "ViewGroup root and attachToRoot=true");
        }
        rInflate(parser, root, inflaterContext, attrs, false);
    } else {
        // 创建根节点的View(也就是经常我们创建的LinearLayout,RelativeLayout)
        final View temp = createViewFromTag(root, name, inflaterContext, attrs);
        ViewGroup.LayoutParams params = null;
        if (root != null) {
            // 生成一个默认匹配root类型的LayoutParams(如:LinearLayout.LayoutParams),且宽高为matchParent
            params = root.generateLayoutParams(attrs);
            if (!attachToRoot) {
                // Set the layout params for temp if we are not
                // attaching. (If we are, we use addView, below)
                temp.setLayoutParams(params);
            }
        }
        // 加载tempView下的所有子View,并add进temp
        rInflateChildren(parser, temp, attrs, true);
        //根据上文提到的,两参数不同,返回不同的result 
        if (root != null && attachToRoot) {
            root.addView(temp, params);
        }
        if (root == null || !attachToRoot) {
            result = temp;
        }
    }
    return result;
}

代码删减了许多,但核心的就是展示了这几个参数对inflate方法的影响。

看了代码,我们发现createViewFromTag方法是根据xml里定义标签生成特定的View,而rInflateChildren是个递归的过程,就是一层层地向下调用createViewFromTag创建子View并且addView到自己的父节点,而createViewFromTag在筛选特定的工程方法(LayoutInflater的子类可以自定义生成View的方式)后,它会调用createView方法,createView才是生成没有子类的View,那它是如何工作的呢。

createView

/**
* 通过LayoutInflater的ClassLoader去实例化给出的name相对于的View
*
*
* @param name View的name
* @param prefix View的前缀,自定义的View为包名,系统的为"android.view."
* @param attrs View的XML attributes
*
* @return View view, 或者null.
*/

    public final View createView(String name, String prefix, AttributeSet attrs)
            throws ClassNotFoundException, InflateException {
        //作为一个View缓冲Map
        Constructor<? extends View> constructor = sConstructorMap.get(name);
        Class<? extends View> clazz = null;

        try {
            //日志追踪
            Trace.traceBegin(Trace.TRACE_TAG_VIEW, name);
            if (constructor == null) {
                // 缓冲为空,加入前缀去加载它
                clazz = mContext.getClassLoader().loadClass(
                        prefix != null ? (prefix + name) : name).asSubclass(View.class);
                
                if (mFilter != null && clazz != null) {
                //View是否允许加载
                    boolean allowed = mFilter.onLoadClass(clazz);
                    if (!allowed) {
                        failNotAllowed(name, prefix, attrs);
                    }
                }
                constructor = clazz.getConstructor(mConstructorSignature);
                constructor.setAccessible(true);
                sConstructorMap.put(name, constructor);
            } else {
                // 缓冲不为空,判断该类能否被加载
                if (mFilter != null) {
                    Boolean allowedState = mFilterMap.get(name);
                    if (allowedState == null) {
                        clazz = mContext.getClassLoader().loadClass(
                                prefix != null ? (prefix + name) : name).asSubclass(View.class);
                        
                        boolean allowed = clazz != null && mFilter.onLoadClass(clazz);
                        mFilterMap.put(name, allowed);
                        if (!allowed) {
                            failNotAllowed(name, prefix, attrs);
                        }
                    } else if (allowedState.equals(Boolean.FALSE)) {
                        failNotAllowed(name, prefix, attrs);
                    }
                }
            }

            Object[] args = mConstructorArgs;
            args[1] = attrs;
            //constructor.newInstance()去返回一个新的View实例
            final View view = constructor.newInstance(args);
            if (view instanceof ViewStub) {
                // 如果该View是ViewStub或者其子类,等它需要加载时使用这个相同的上下文
                final ViewStub viewStub = (ViewStub) view;
                viewStub.setLayoutInflater(cloneInContext((Context) args[0]));
            }
            return view;

        } catch (NoSuchMethodException e) {
            InflateException ie = new InflateException(attrs.getPositionDescription()
                    + ": Error inflating class "
                    + (prefix != null ? (prefix + name) : name));
            ie.initCause(e);
            throw ie;

        } catch (ClassCastException e) {
            // If loaded class is not a View subclass
            InflateException ie = new InflateException(attrs.getPositionDescription()
                    + ": Class is not a View "
                    + (prefix != null ? (prefix + name) : name));
            ie.initCause(e);
            throw ie;
        } catch (ClassNotFoundException e) {
            // If loadClass fails, we should propagate the exception.
            throw e;
        } catch (Exception e) {
            InflateException ie = new InflateException(attrs.getPositionDescription()
                    + ": Error inflating class "
                    + (clazz == null ? "<unknown>" : clazz.getName()));
            ie.initCause(e);
            throw ie;
        } finally {
            Trace.traceEnd(Trace.TRACE_TAG_VIEW);
        }
    }

createView调用完成后,递归就会往回走,又回到刚才的rInflateChildrencreateViewFromTag下一个子View,当所有的View都加载完成了,就执行上文提到的inflate方法里的root.addView(temp, params),最后return result
OK,整个inflate过程就结束了。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容