View_01_LayoutInflater的原理、使用方法

本篇博客是郭神博客Android视图状态及重绘流程分析,带你一步步深入了解View(一)的读书笔记的笔记。

LayoutInflater简单介绍

setContentView()内部是使用LayoutInflater来完毕载入布局的。

setContentView()方法内部的源代码是internal的。不太easy查到。
翻了七八个类后,在AppCompatDelegateImplV7.java中找到了这段代码

@Override
public void setContentView(int resId) {ensureSubDecor();ViewGroup contentParent = (ViewGroup) mSubDecor.findViewById(android.R.id.content);contentParent.removeAllViews();LayoutInflater.from(mContext).inflate(resId, contentParent);mOriginalWindowCallback.onContentChanged();
}

获取LayoutInflater实例

第一种方法:

     LayoutInflater layoutInflater = LayoutInflater.from(context);

另外一种方法:

     LayoutInflater layoutInflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);

事实上另外一种方法是第一种的简单写法,仅仅是Android给我们做了封装而已。

layoutInflater.inflate()方法

演示样例:

layoutInflater.inflate(resourceId, root);

第一个參数就是要载入的布局id,第二个參数是指给该布局的外部再嵌套一层父布局,假设不须要就直接传null。这样就成功成功创建了一个布局的实例。之后再将它加入到指定的位置就能够显示出来了。

实例:LayoutInflater的使用方法

activity_main.xml

 <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:id="@+id/main_layout"android:layout_width="match_parent"android:layout_height="match_parent" ></LinearLayout>

button_layout.xml

 <Button xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="Button" ></Button>

如今我们要想办法,怎样通过LayoutInflater来将button_layout这个布局加入到主布局文件的LinearLayout中。依据刚刚介绍的使用方法,改动MainActivity中的代码,例如以下所看到的:

MainActivity.java

public class MainActivity extends Activity {private LinearLayout mainLayout;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);mainLayout = (LinearLayout) findViewById(R.id.main_layout);LayoutInflater layoutInflater = LayoutInflater.from(this);View buttonLayout = layoutInflater.inflate(R.layout.button_layout, null);mainLayout.addView(buttonLayout);}}

结果例如以下图所看到的

LayoutInflater技术广泛应用于须要动态加入View的时候,比方在ScrollView和ListView中,常常都能够看到LayoutInflater的身影。

LayoutInflater的工作原理

inflate源代码

public View inflate(XmlPullParser parser, ViewGroup root, boolean attachToRoot) {synchronized (mConstructorArgs) {final AttributeSet attrs = Xml.asAttributeSet(parser);mConstructorArgs[0] = mContext;View result = root;try {int type;while ((type = parser.next()) != XmlPullParser.START_TAG &&type != XmlPullParser.END_DOCUMENT) {}if (type != XmlPullParser.START_TAG) {throw new InflateException(parser.getPositionDescription()+ ": No start tag found!");}final String name = parser.getName();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, attrs);} else {View temp = createViewFromTag(name, attrs);ViewGroup.LayoutParams params = null;if (root != null) {params = root.generateLayoutParams(attrs);if (!attachToRoot) {temp.setLayoutParams(params);}}rInflate(parser, temp, attrs);if (root != null && attachToRoot) {root.addView(temp, params);}if (root == null || !attachToRoot) {result = temp;}}} catch (XmlPullParserException e) {InflateException ex = new InflateException(e.getMessage());ex.initCause(e);throw ex;} catch (IOException e) {InflateException ex = new InflateException(parser.getPositionDescription()+ ": " + e.getMessage());ex.initCause(e);throw ex;}return result;}
}

从这里我们就能够清楚地看出,LayoutInflater事实上就是使用Android提供的pull解析方式来解析布局文件的。
这里我们注意看下第23行,调用了createViewFromTag()这种方法。并把节点名和參数传了进去。

看到这种方法名。我们就应该能猜到。它是用于依据节点名来创建View对象的。确实如此,在createViewFromTag()方法的内部又会去调用createView()方法,然后使用反射的方式创建出View的实例并返回。
当然,这里仅仅是创建出了一个根布局的实例而已。接下来会在第31行调用rInflate()方法来循环遍历这个根布局下的子元素,代码例如以下所看到的:

rInflate源代码

private void rInflate(XmlPullParser parser, View parent, final AttributeSet attrs)throws XmlPullParserException, IOException {final int depth = parser.getDepth();int type;while (((type = parser.next()) != XmlPullParser.END_TAG ||parser.getDepth() > depth) && type != XmlPullParser.END_DOCUMENT) {if (type != XmlPullParser.START_TAG) {continue;}final String name = parser.getName();if (TAG_REQUEST_FOCUS.equals(name)) {parseRequestFocus(parser, parent);} else if (TAG_INCLUDE.equals(name)) {if (parser.getDepth() == 0) {throw new InflateException("<include /> cannot be the root element");}parseInclude(parser, parent, attrs);} else if (TAG_MERGE.equals(name)) {throw new InflateException("<merge /> must be the root element");} else {final View view = createViewFromTag(name, attrs);final ViewGroup viewGroup = (ViewGroup) parent;final ViewGroup.LayoutParams params = viewGroup.generateLayoutParams(attrs);rInflate(parser, view, attrs);viewGroup.addView(view, params);}}parent.onFinishInflate();
}

这种话。把整个布局文件都解析完毕后就形成了一个完整的DOM结构。终于会把最顶层的根布局返回,至此inflate()过程所有结束。
比較细心的朋友或许会注意到,inflate()方法还有个接收三个參数的方法重载,结构例如以下:

inflate()方法的第三个參数

inflate(int resource, ViewGroup root, boolean attachToRoot)
1. 假设root为null。attachToRoot将失去作用。设置不论什么值都没有意义。
2. 假设root不为null,attachToRoot设为true,则会给载入的布局文件的指定一个父布局,即root。
3. 假设root不为null,attachToRoot设为false,则会将布局文件最外层的所有layout属性进行设置,当该view被加入到父view其中时,这些layout属性会自己主动生效。

4. 在不设置attachToRoot參数的情况下,假设root不为null,attachToRoot參数默觉得true。

将上面实例中的Button按钮变大

事实上这里无论你将Button的layout_width和layout_height的值改动成多少。都不会有不论什么效果的,由于这两个值如今已经全然失去了作用。

平时我们常常使用layout_width和layout_height来设置View的大小,而且一直都能正常工作。就好像这两个属性确实是用于设置View的大小的。

而实际上则不然,它们事实上是用于设置View在布局中的大小的,也就是说,首先View必须存在于一个布局中,之后假设将layout_width设置成match_parent表示让View的宽度填充满布局,假设设置成wrap_content表示让View的宽度刚好能够包括其内容,假设设置成详细的数值则View的宽度会变成对应的数值。这也是为什么这两个属性叫作layout_width和layout_height,而不是width和height。

改动button的布局文件。不能变大

<Button xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="300dp"android:layout_height="80dp"android:text="Button" >
</Button>

在button外嵌套一层布局,可使其变大

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="match_parent"android:layout_height="match_parent" ><Button
        android:layout_width="300dp"android:layout_height="80dp"android:text="Button" ></Button>
</RelativeLayout>

能够看到,这里我们又加入了一个RelativeLayout,此时的Button存在与RelativeLayout之中。layout_width和layout_height属性也就有作用了。

当然,处于最外层的RelativeLayout,它的layout_width和layout_height则会失去作用。如今又一次执行一下程序,结果例如以下图所看到的

setContentView()为什么叫setContentView ?

看到这里,或许有些朋友心中会有一个巨大的疑惑。不正确呀!平时在Activity中指定布局文件的时候,最外层的那个布局是能够指定大小的呀,layout_width和layout_height都是有作用的。

确实,这主要是由于。在setContentView()方法中,Android会自己主动在布局文件的最外层再嵌套一个FrameLayout。所以layout_width和layout_height属性才会有效果。那么我们来证实一下吧。改动MainActivity中的代码,例如以下所看到的:

MainActivity.java

public class MainActivity extends Activity {private LinearLayout mainLayout;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);mainLayout = (LinearLayout) findViewById(R.id.main_layout);ViewParent viewParent = mainLayout.getParent();Log.d("TAG", "the parent of mainLayout is " + viewParent);}}

打印结果:

LinearLayout的父布局确实是一个FrameLayout,而这个FrameLayout就是由系统自己主动帮我们加入上的。
讲到这里。尽管setContentView()方法大家都会用,但实际上Android界面显示的原理要比我们所看到的东西复杂得多。不论什么一个Activity中显示的界面事实上主要都由两部分组成,标题栏和内容布局。

标题栏就是在非常多界面顶部显示的那部分内容,比方刚刚我们的那个样例其中就有标题栏,能够在代码中控制让它是否显示。

而内容布局就是一个FrameLayout,这个布局的id叫作content,我们调用setContentView()方法时所传入的布局事实上就是放到这个FrameLayout中的。这也是为什么这种方法名叫作setContentView(),而不是叫setView()。
最后再附上一张Activity窗体的组成图吧,以便于大家更加直观地理解:

小结

本篇主要介绍了LayoutInflater的原理、使用方法,为深入理解View做了准备。

作者信息

  • 个人博客:http://blog.csdn.net/fy9987899
  • github项目:https://github.com/fanyu2013/Android-Blogs

View_01_LayoutInflater的原理、使用方法相关推荐

  1. c++程序设计原理与实践_课程思政水资源系统优化原理与方法课程思政元素的探索...

    案例说明+课程基本信息 案例说明: 水资源系统优化原理与方法的思政教育目标是要让学生建立一种科学思维方式,用辩证和历史唯物主义的观点去观察和分析问题,培养其规则意识和约束观念,以社会主义核心价值观来进 ...

  2. 中文路径_中文分词的原理、方法与工具

    海德格尔说"词语破碎处,无物可存在".中文句子不像英文那样的词与词之间有显示空格边界,使得词和词组边界模糊. 为了让计算机更容易理解文本,通常中文信息处理的第一步是中文分词.中文分 ...

  3. x12arima季节调整方法_《时间序列X-12-ARIMA季节调整:原理与方法》

    对时间序列进行季节调整是经济分析的基础性工作.人民银行组织力量对季节调整基本方法进行了研究,结合调整中国特有的移动假日--春节的需要,提出了不同的处理模型,对各国较为通用的季节调整软件X-12-ARI ...

  4. 《异构信息网络挖掘: 原理和方法(1)》一第2章 基于排名的聚类

    本节书摘来自华章出版社<异构信息网络挖掘: 原理和方法(1)>一书中的第2章,作者[美]孙艺洲(Yizhou Sun)韩家炜(Jiawei Han),更多章节内容可以访问云栖社区" ...

  5. linux怎么查看数据库性能,正确评估SQL数据库性能,你必须知道的原理和方法!...

    原标题:正确评估SQL数据库性能,你必须知道的原理和方法! 作者:阿特 来源: http://blog.csdn.net/capsicum29/article/details/71480799 数据库 ...

  6. 正确评估SQL数据库性能,你必须知道的原理和方法!

    作者简介: Max Shen(阿特),为了成为数据专家而努力,万一实现了呢! 昨天写了一篇如何监视数据库性能,了解数据库的运行状态.被有人质疑,说没有用.说要直接用数据库的profile和monito ...

  7. 链式调用方法的实现原理和方法

    1.什么是链式调用? Person person = new Person().setName(fog).setAge(18).setSex(man).setJob(software engineer ...

  8. 数字测图原理与方法的实习日志_数字测图原理与方法实习与习题.doc

    数字测图原理与方法实习与习题.doc 习题第一章测量的基本知识一.问答题1.什么是水准面它有什么特性2.什么是大地水准面它有什么特性它在测量工作中起什么作用3.测量工作中为什么选取与大地体相近的旋转椭 ...

  9. 教你把gps服务器修改为中国加速搜星,Android的GPS加速搜星的原理和方法

    把默认的北美NTP_SERVER以及SUPL_HOST修改为中国区的,当然,亚洲区也行,只要是网速快的就行. Android系统GPS加速搜星的原理和方法,修改GPS定位服务器为中国的 本文来自网络, ...

最新文章

  1. 《需求分析与系统设计》阅读笔记三
  2. Android 实现歌曲播放时歌词同步显示
  3. 支持常见数据库差异对照说明
  4. 机器人学习--栅格地图(occupancy grid map)构建
  5. Scala _03方法与函数
  6. wordpress文章添加css样式,给WordPress文章循环加上CSS类方便实现各种页面布局
  7. 开发进阶:Dotnet Core多路径异步终止
  8. java rx.observable_Rxjava2 Observable的条件操作符详解及实例
  9. HAproxy部署配置
  10. VXWORKS 几种定时机制
  11. 浅析jQuery中常用的元素查找方法总结
  12. Linux中如何使用帮助
  13. 查看,添加和删除GIT配置的正确姿势
  14. ECMAScript——(二)
  15. 曙光超级计算机用的芯片是国产吗,真正中国芯片龙头是中科曙光
  16. 如何在PowerPoint中更改文本的大小写
  17. vue下载excel表格模板和导入excel表格数据
  18. 互联网日报 | 7月13日 星期二 | 张近东辞任苏宁易购董事长;斗鱼虎牙宣布终止合并;联想集团二季度蝉联全球PC销量冠军...
  19. python转盘抽奖_python实现转盘效果 python实现轮盘抽奖游戏
  20. 什么是全量表,增量表,快照表,拉链表?

热门文章

  1. 深度学习中的验证集和超参数简介
  2. OpenCV中SVM的使用
  3. 当代的设计潮流是什么_解码“潮流合伙人”IP生意经
  4. linux特殊系统变量,linux环境几个特殊的shell变量
  5. 安卓蓝牙键盘按键映射_双层按键功能自定义:魔蛋68蓝牙双模机械键盘体验
  6. SpringBoot复习:2(@Configuration注解)
  7. android首页图片轮播效果,Android_Android自动播放Banner图片轮播效果,先看一下效果图支持本地图 - phpStudy...
  8. arduino 控制无刷电机_智能控制轮椅来了,残疾人的福音!
  9. Angular应用中配置全局路径映射
  10. Android 常见工具类封装