Android中LayoutInflater.inflater()的正確打開方式
前言
LayoutInflater在開發(fā)中使用頻率很高,但是一直沒有太知道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所對應(yīng)的系統(tǒng)服務(wù)。這里涉及到Context實現(xiàn)以及服務(wù)創(chuàng)建的源碼,不繼續(xù)深究。
重點是通常所使用的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) ,
而兩個參數(shù)的方法,實際也是調(diào)用了三個參數(shù)的inflate()方法,只是在三個參數(shù)傳入了root!=null
public View inflate(@LayoutRes int resource, @Nullable ViewGroup root) {
return inflate(resource, root, root != null);
}
那我們就可以直接看三個參數(shù)的inflate()方法了,其中res.getLayout(resource)這句代碼,已經(jīng)將我們傳入的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();
}
}
這里其實就會發(fā)現(xiàn),最后return調(diào)用的其實是inflate(XmlPullParser parser, @Nullable ViewGroup root, boolean attachToRoot)這個方法,所謂的四個inflate()方法,其他三個只是對這個方法的重載,主要代碼還是在這個方法中實現(xiàn)的
這部分代碼較長,以注釋的形式解釋代碼
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對象轉(zhuǎn)換成AttributeSet
final AttributeSet attrs = Xml.asAttributeSet(parser);
Context lastContext = (Context) mConstructorArgs[0];
mConstructorArgs[0] = inflaterContext;
View result = root;
try {
//2.在xml中尋找根節(jié)點,如果類型是XmlPullParser.START_TAG或者XmlPullParser.END_DOCUMENT就會退出循環(huán)
int type;
while ((type = parser.next()) != XmlPullParser.START_TAG &&
type != XmlPullParser.END_DOCUMENT) {
// Empty
}
//3.如果根節(jié)點類型不是XmlPullParser.START_TAG將拋出異常
if (type != XmlPullParser.START_TAG) {
throw new InflateException(parser.getPositionDescription()
+ ": No start tag found!");
}
final String name = parser.getName();
//4.判斷根節(jié)點是否是merge標(biāo)簽
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.通過根節(jié)點創(chuàng)建臨時的view對象
final View temp = createViewFromTag(root, name, inflaterContext, attrs);
ViewGroup.LayoutParams params = null;
if (root != null) {
//6.如果root不為空,則調(diào)用generateLayoutParams(attrs)獲取root所對應(yīng)LayoutParams對象
params = root.generateLayoutParams(attrs);
//是否attachToRoot
if (!attachToRoot) {
//7.如果attachToRoot為false,則使用root默認(rèn)的LayoutParams作為臨時view對象的屬性
temp.setLayoutParams(params);
}
}
//8.inflate xml的所有子節(jié)點
rInflateChildren(parser, temp, attrs, true);
//9.判斷是否需要將創(chuàng)建的臨時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中跟布局的屬性數(shù)據(jù)。
6,7則是很多時候使用inflate方法之后,發(fā)現(xiàn)xml布局設(shè)置的寬高屬性不生效的部分原因,有時候在RecyclerView中添加就會這樣。如果root!=null且attachToRoot為false時,創(chuàng)建的view則會具有自身根節(jié)點屬性值,與root對應(yīng)的LayoutParam
9的判斷決定了創(chuàng)建的view是否添加到root中,而10則決定了方法返回的是root還是view
總結(jié)
根據(jù)inflate的參數(shù)不同可以獲得不同的返回值
| root | attachToRoot | 返回值 |
|---|---|---|
| null | false(或者true) | 返回resource對應(yīng)的view對象,但是xml中根節(jié)點的屬性沒有生效 |
| !=null | false | 返回resource對應(yīng)的view對象,并且xml中根節(jié)點的屬性生效,view對象的LayoutParam與root的LayoutParam對應(yīng) |
| !=null | true | 返回root對象,對應(yīng)resource創(chuàng)建的view對象,xml中根節(jié)點的屬性生效,并且將會添加到root中 |
注意:attachToRoot默認(rèn)為root!=null的值
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
相關(guān)文章
詳解Android 通過Socket 和服務(wù)器通訊(附demo)
Android 通過Socket 和服務(wù)器通訊,是一種比較常用的通訊方式,這篇文章主要介紹了詳解Android 通過Socket 和服務(wù)器通訊,有興趣的可以了解一下。2016-12-12
Flutter實現(xiàn)資源下載斷點續(xù)傳的示例代碼
在項目開發(fā)中,特別是C端的產(chǎn)品,資源下載實現(xiàn)斷點續(xù)傳是非常有必要的。今天我們不講過多原理的知識,分享下簡單實用的資源斷點續(xù)傳2022-07-07
Flutter 使用cached_image_network優(yōu)化圖片加載體驗
在 Flutter 中,cached_image_network 即提供了緩存網(wǎng)絡(luò)圖片功能,同時還提供了豐富的加載過程指示。本文就來看下cached_image_network的具體使用2021-05-05
Eclipse工程轉(zhuǎn)為兼容Android Studio模式的方法步驟圖文詳解
這篇文章主要介紹了Eclipse工程轉(zhuǎn)為兼容Android Studio模式的方法步驟,本文圖文并茂給大家介紹的非常詳細(xì),需要的朋友可以參考下2017-12-12

