這篇文章是學(xué)習(xí)郭大神的系列文章的筆記,原文在下面
Android LayoutInflater原理分析收津,帶你一步步深入了解View(一)
-
首先需要獲取到LayoutInflater的實(shí)例伙窃,
LayoutInflater layoutInflater = LayoutInflater.from(context);
LayoutInflater layoutInflater = (LayoutInflater) context .getSystemService(Context.LAYOUT_INFLATER_SERVICE);
-
layoutInflater.inflate(resourceId, root);
inflate()方法一般接收兩個(gè)參數(shù)菩颖,第一個(gè)參數(shù)就是要加載的布局id,第二個(gè)參數(shù)是指給該布局的外部再嵌套一層父布局为障,如果不需要就直接傳null晦闰。這樣就成功成功創(chuàng)建了一個(gè)布局的實(shí)例,之后再將它添加到指定的位置就可以顯示出來了鳍怨。
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其實(shí)就是使用Android提供的pull解析方式來解析布局文件的呻右。
* 它們其實(shí)是用于設(shè)置View在布局中的大小的,也就是說鞋喇,首先View必須存在于一個(gè)布
局中声滥,之后如果將layout_width設(shè)置成match_parent表示讓View的寬度填充滿布局,如果設(shè)置成wrap_content表示讓View的寬度剛好可以包含其內(nèi)容侦香,如果設(shè)置成具體的數(shù)值則View的寬度會變成相應(yīng)的數(shù)值落塑。這也是為什么這兩個(gè)屬性叫作layout_width和layout_height纽疟,而不是width和height
但是在我們平時(shí)setContentView的時(shí)候,我們并沒有在外指定一個(gè)父布局憾赁,但是仍然起效污朽,是因?yàn)橄到y(tǒng)在setContentView之前自動幫我們指定了一個(gè)FrameLayout.
![contentView](http://upload-images.jianshu.io/upload_images/1859111-a5c5c912e6e5495d?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
![View樹結(jié)構(gòu)](http://upload-images.jianshu.io/upload_images/1859111-e5c2d8b979b0f04b.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
每個(gè)Activity都包含一個(gè)Window對象,在android中Window對象通常有phoneWindow來實(shí)現(xiàn)龙考,PhoneWindow將一個(gè)DecorView設(shè)置為整個(gè)應(yīng)用的窗口的根View蟆肆。
ContentView就是一個(gè)ID為content的FrameLayout,activity_main.xml就是設(shè)置在這樣的一個(gè)FrameLayout中。
![UI界面架構(gòu)圖](http://upload-images.jianshu.io/upload_images/1859111-b4ce7638d58b1536.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)