Android中LayoutInflater.inflater()的正确打开方式

所属分类: 软件编程 / Android 阅读数: 102
收藏 0 赞 0 分享

前言

LayoutInflater在开发中使用频率很高,但是一直没有太知道LayoutInflater.from(context).inflate()的真正用法,今天就看看源码的流程。

首先来看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 new AssertionError("LayoutInflater not found.");
 }
 return LayoutInflater;
}

其实就是从Context中获取Context.LAYOUT_INFLATER_SERVICE所对应的系统服务。这里涉及到Context实现以及服务创建的源码,不继续深究。

重点是通常所使用的inflate()方法,比较常用的就是这两个:

  • inflate(@LayoutRes int resource, @Nullable ViewGroup root)
  • inflate(@LayoutRes int resource, @Nullable ViewGroup root, boolean attachToRoot)

另外两个方法inflate(XmlPullParser parser, @Nullable ViewGroup root)inflate(XmlPullParser parser, @Nullable ViewGroup root, boolean attachToRoot)

而两个参数的方法,实际也是调用了三个参数的inflate()方法,只是在三个参数传入了root!=null

public View inflate(@LayoutRes int resource, @Nullable ViewGroup root) {
 return inflate(resource, root, root != null);
}

那我们就可以直接看三个参数的inflate()方法了,其中res.getLayout(resource)这句代码,已经将我们传入的layout布局的根布局的xml属性都加载到了XmlResourceParser中

public View inflate(@LayoutRes int resource, @Nullable ViewGroup root, boolean attachToRoot) {
 final Resources res = getContext().getResources();
 //省略代码
 final XmlResourceParser parser = res.getLayout(resource);
 try {
 return inflate(parser, root, attachToRoot);
 } finally {
 parser.close();
 }
}

这里其实就会发现,最后return调用的其实是inflate(XmlPullParser parser, @Nullable ViewGroup root, boolean attachToRoot)这个方法,所谓的四个inflate()方法,其他三个只是对这个方法的重载,主要代码还是在这个方法中实现的

这部分代码较长,以注释的形式解释代码

public View inflate(XmlPullParser parser, @Nullable ViewGroup root, boolean attachToRoot) {
 synchronized (mConstructorArgs) {
 Trace.traceBegin(Trace.TRACE_TAG_VIEW, "inflate");

 final Context inflaterContext = mContext;
 //1.通过XmlResourceParser对象转换成AttributeSet
 final AttributeSet attrs = Xml.asAttributeSet(parser);
 Context lastContext = (Context) mConstructorArgs[0];
 mConstructorArgs[0] = inflaterContext;
 View result = root;

 try {
  //2.在xml中寻找根节点,如果类型是XmlPullParser.START_TAG或者XmlPullParser.END_DOCUMENT就会退出循环
  int type;
  while ((type = parser.next()) != XmlPullParser.START_TAG &&
   type != XmlPullParser.END_DOCUMENT) {
  // Empty
  }
  //3.如果根节点类型不是XmlPullParser.START_TAG将抛出异常
  if (type != XmlPullParser.START_TAG) {
  throw new InflateException(parser.getPositionDescription()
   + ": No start tag found!");
  }

  final String name = parser.getName();

  //4.判断根节点是否是merge标签
  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 {
  //5.通过根节点创建临时的view对象
  final View temp = createViewFromTag(root, name, inflaterContext, attrs);

  ViewGroup.LayoutParams params = null;

  if (root != null) {
   //6.如果root不为空,则调用generateLayoutParams(attrs)获取root所对应LayoutParams对象
   params = root.generateLayoutParams(attrs);
   //是否attachToRoot
   if (!attachToRoot) {
   //7.如果attachToRoot为false,则使用root默认的LayoutParams作为临时view对象的属性
   temp.setLayoutParams(params);
   }
  }

  //8.inflate xml的所有子节点
  rInflateChildren(parser, temp, attrs, true);

  //9.判断是否需要将创建的临时view attach到root中
  if (root != null && attachToRoot) {
   root.addView(temp, params);
  }

  //10.决定方法的返回值是root还是临时view
  if (root == null || !attachToRoot) {
   result = temp;
  }
  }

 } catch (XmlPullParserException e) {
  final InflateException ie = new InflateException(e.getMessage(), e);
  ie.setStackTrace(EMPTY_STACK_TRACE);
  throw ie;
 } catch (Exception e) {
  final InflateException ie = new InflateException(parser.getPositionDescription()
   + ": " + e.getMessage(), e);
  ie.setStackTrace(EMPTY_STACK_TRACE);
  throw ie;
 } finally {
  mConstructorArgs[0] = lastContext;
  mConstructorArgs[1] = null;

  Trace.traceEnd(Trace.TRACE_TAG_VIEW);
 }

 return result;
 }
}

1中的XmlResourceParser在之前所获取的,包含了layout中跟布局的属性数据。

6,7则是很多时候使用inflate方法之后,发现xml布局设置的宽高属性不生效的部分原因,有时候在RecyclerView中添加就会这样。如果root!=null且attachToRoot为false时,创建的view则会具有自身根节点属性值,与root对应的LayoutParam

9的判断决定了创建的view是否添加到root中,而10则决定了方法返回的是root还是view

总结

根据inflate的参数不同可以获得不同的返回值

root attachToRoot 返回值
null false(或者true) 返回resource对应的view对象,但是xml中根节点的属性没有生效
!=null false 返回resource对应的view对象,并且xml中根节点的属性生效,view对象的LayoutParam与root的LayoutParam对应
!=null true 返回root对象,对应resource创建的view对象,xml中根节点的属性生效,并且将会添加到root中

注意:attachToRoot默认为root!=null的值

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对脚本之家的支持。

更多精彩内容其他人还在看

Android网络编程之获取网络上的Json数据实例

这篇文章主要介绍了Android网络编程之获取网络上的Json数据实例,本文用完整的代码实例讲解了在Android中读取网络中Json数据的方法,需要的朋友可以参考下
收藏 0 赞 0 分享

Android中的windowSoftInputMode属性详解

这篇文章主要介绍了Android中的windowSoftInputMode属性详解,本文对windowSoftInputMode的9个属性做了详细总结,需要的朋友可以参考下
收藏 0 赞 0 分享

Android网络编程之UDP通信模型实例

这篇文章主要介绍了Android网络编程之UDP通信模型实例,本文给出了服务端代码和客户端代码,需要的朋友可以参考下
收藏 0 赞 0 分享

Android中使用ListView实现漂亮的表格效果

这篇文章主要介绍了Android中使用ListView实现漂亮的表格效果,本文用详细的代码实例创建了一个股票行情表格,需要的朋友可以参考下
收藏 0 赞 0 分享

Android中刷新界面的二种方法

这篇文章主要介绍了Android中刷新界面的二种方法,本文使用Handler、postInvalidate两种方法实现界面刷新,需要的朋友可以参考下
收藏 0 赞 0 分享

Android SDK三种更新失败及其解决方法

这篇文章主要介绍了Android SDK三种更新失败及其解决方法,需要的朋友可以参考下
收藏 0 赞 0 分享

Android学习笔记——Menu介绍(一)

Android3.0(API level 11)开始,Android设备不再需要专门的菜单键。随着这种变化,Android app应该取消对传统6项菜单的依赖。取而代之的是提供anction bar来提供基本的用户功能
收藏 0 赞 0 分享

Android学习笔记——Menu介绍(二)

这次将继续上一篇文章没有讲完的Menu的学习,上下文菜单(Context menu)和弹出菜单(Popup menu)
收藏 0 赞 0 分享

Android学习笔记——Menu介绍(三)

今天继续昨天没有讲完的Menu的学习,主要是Popup Menu的学习,需要的朋友可以参考下
收藏 0 赞 0 分享

Android显示网络图片实例

这篇文章主要介绍了Android显示网络图片的方法,以实例形式展示了Android程序显示网络图片的方法,非常具有实用价值,需要的朋友可以参考下
收藏 0 赞 0 分享
查看更多