这篇文章是学习郭大神的系列文章的笔记,原文在下面
Android LayoutInflater原理分析,带你一步步深入了解View(一)
-
首先需要获取到LayoutInflater的实例,
LayoutInflater layoutInflater = LayoutInflater.from(context);
LayoutInflater layoutInflater = (LayoutInflater) context .getSystemService(Context.LAYOUT_INFLATER_SERVICE);
-
layoutInflater.inflate(resourceId, root);
inflate()方法一般接收两个参数,第一个参数就是要加载的布局id,第二个参数是指给该布局的外部再嵌套一层父布局,如果不需要就直接传null。这样就成功成功创建了一个布局的实例,之后再将它添加到指定的位置就可以显示出来了。
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解析方式来解析布局文件的。
* 它们其实是用于设置View在布局中的大小的,也就是说,首先View必须存在于一个布
局中,之后如果将layout_width设置成match_parent表示让View的宽度填充满布局,如果设置成wrap_content表示让View的宽度刚好可以包含其内容,如果设置成具体的数值则View的宽度会变成相应的数值。这也是为什么这两个属性叫作layout_width和layout_height,而不是width和height
但是在我们平时setContentView的时候,我们并没有在外指定一个父布局,但是仍然起效,是因为系统在setContentView之前自动帮我们指定了一个FrameLayout.
![contentView](http://upload-images.jianshu.io/upload_images/1859111-a5c5c912e6e5495d?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
![View树结构](http://upload-images.jianshu.io/upload_images/1859111-e5c2d8b979b0f04b.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
每个Activity都包含一个Window对象,在android中Window对象通常有phoneWindow来实现,PhoneWindow将一个DecorView设置为整个应用的窗口的根View。
ContentView就是一个ID为content的FrameLayout,activity_main.xml就是设置在这样的一个FrameLayout中。
![UI界面架构图](http://upload-images.jianshu.io/upload_images/1859111-b4ce7638d58b1536.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)