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

作者:MrTrying 时间:2022-04-12 16:25:57 

前言

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的参数不同可以获得不同的返回值

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


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

来源:https://www.jianshu.com/p/b235b2516d4f

标签:android,layoutInflater.inflater()
0
投稿

猜你喜欢

  • 字符串替换Replace仅替换第一个字符串匹配项

    2021-10-02 17:36:56
  • Android实现文字垂直滚动、纵向走马灯效果的实现方式汇总

    2023-03-02 22:24:54
  • Java读取txt文件和写入txt文件的简单实例

    2022-01-01 05:04:16
  • 利用Jackson解决Json序列化和反序列化问题

    2023-02-16 14:59:54
  • JAVA JVM面试题总结

    2021-07-12 04:55:13
  • c# winform异步不卡界面的实现方法

    2021-10-03 22:27:33
  • 常见JavaWeb安全问题和解决方案

    2023-11-09 01:02:17
  • C#编程实现动态改变配置文件信息的方法

    2022-05-01 15:09:25
  • 自定义spring mvc的json视图实现思路解析

    2023-03-12 12:04:44
  • 使用java的HttpClient实现多线程并发

    2022-09-19 20:35:55
  • 详解Glide4.0集成及使用注意事项

    2021-12-28 00:09:06
  • Flutter WillPopScope拦截返回事件原理示例详解

    2023-07-19 12:53:35
  • 简单实现Android闹钟功能

    2022-05-23 16:17:53
  • Java多线程之读写锁分离设计模式

    2021-06-08 07:20:48
  • java实现订餐系统

    2023-08-12 04:39:06
  • 详解SpringBoot2.0的@Cacheable(Redis)缓存失效时间解决方案

    2023-07-23 07:20:54
  • 深入了解Java虚拟机栈以及内存模型

    2022-02-17 13:37:50
  • Java的JDBC中Statement与CallableStatement对象实例

    2022-11-14 17:22:08
  • 深入了解Hadoop如何实现序列化

    2023-10-13 10:33:43
  • Mybatis 入参类型方式全面详解

    2023-10-16 20:03:40
  • asp之家 软件编程 m.aspxhome.com