如何在Android中使用LayoutInflater.inflater()?相信很多沒有經(jīng)驗的人對此束手無策,為此本文總結了問題出現(xiàn)的原因和解決方法,通過這篇文章希望你能解決這個問題。
創(chuàng)新互聯(lián)建站專注于企業(yè)成都全網(wǎng)營銷、網(wǎng)站重做改版、丹寨網(wǎng)站定制設計、自適應品牌網(wǎng)站建設、H5網(wǎng)站設計、商城網(wǎng)站建設、集團公司官網(wǎng)建設、成都外貿網(wǎng)站建設公司、高端網(wǎng)站制作、響應式網(wǎng)頁設計等建站業(yè)務,價格優(yōu)惠性價比高,為丹寨等各大城市提供網(wǎng)站開發(fā)制作服務。
首先來看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所對應的系統(tǒng)服務。這里涉及到Context實現(xiàn)以及服務創(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ù)的方法,實際也是調用了三個參數(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調用的其實是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對象轉換成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標簽 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不為空,則調用generateLayoutParams(attrs)獲取root所對應LayoutParams對象 params = root.generateLayoutParams(attrs); //是否attachToRoot if (!attachToRoot) { //7.如果attachToRoot為false,則使用root默認的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布局設置的寬高屬性不生效的部分原因,有時候在RecyclerView中添加就會這樣。如果root!=null
且attachToRoot為false時,創(chuàng)建的view則會具有自身根節(jié)點屬性值,與root對應的LayoutParam
9的判斷決定了創(chuàng)建的view是否添加到root中,而10則決定了方法返回的是root還是view
總結
根據(jù)inflate的參數(shù)不同可以獲得不同的返回值
root | attachToRoot | 返回值 |
---|---|---|
null | false(或者true) | 返回resource對應的view對象,但是xml中根節(jié)點的屬性沒有生效 |
!=null | false | 返回resource對應的view對象,并且xml中根節(jié)點的屬性生效,view對象的LayoutParam與root的LayoutParam對應 |
!=null | true | 返回root對象,對應resource創(chuàng)建的view對象,xml中根節(jié)點的屬性生效,并且將會添加到root中 |
注意:attachToRoot默認為root!=null
的值
看完上述內容,你們掌握如何在Android中使用LayoutInflater.inflater()的方法了嗎?如果還想學到更多技能或想了解更多相關內容,歡迎關注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝各位的閱讀!
本文標題:如何在Android中使用LayoutInflater.inflater()
分享URL:http://jinyejixie.com/article26/ghhsjg.html
成都網(wǎng)站建設公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站內鏈、App設計、ChatGPT、小程序開發(fā)、營銷型網(wǎng)站建設、做網(wǎng)站
聲明:本網(wǎng)站發(fā)布的內容(圖片、視頻和文字)以用戶投稿、用戶轉載內容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內容未經(jīng)允許不得轉載,或轉載時需注明來源: 創(chuàng)新互聯(lián)