a亚洲精品_精品国产91乱码一区二区三区_亚洲精品在线免费观看视频_欧美日韩亚洲国产综合_久久久久久久久久久成人_在线区

首頁 > 系統(tǒng) > Android > 正文

Android中LayoutInflater.inflater()的正確打開方式

2019-10-21 21:34:06
字體:
供稿:網(wǎng)友

前言

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í)價值,如果有疑問大家可以留言交流,謝謝大家對VEVB武林網(wǎng)的支持。


注:相關(guān)教程知識閱讀請移步到Android開發(fā)頻道。
發(fā)表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發(fā)表
主站蜘蛛池模板: 美女黄色在线观看 | 人人草天天草 | 91精品久久久久久久久 | 青娱乐在线播放 | 久久99国产精品久久99大师 | 国产一级在线观看 | 色网站免费 | 伊人一区 | 国产精品亚洲一区二区三区 | 午夜精品福利一区二区三区蜜桃 | 亚洲欧美综合精品久久成人 | 日韩欧美国产精品 | 亚洲精品9999 | 精品视频二区三区 | 国产中文字幕在线 | 精品永久 | 一级毛片观看 | 日本福利一区 | 欧美日韩一区二区三区在线观看 | www97影院| 国产区免费 | 亚洲成av人片一区二区三区 | 久久精品一区二区三区不卡牛牛 | 久久久久91| 欧美成人小视频 | 大陆一级毛片免费看 | 国产人久久人人人人爽 | av影院在线 | 北条麻妃一区二区三区在线观看 | 亚洲美女一区二区三区 | 亚洲久久 | 不卡一区二区三区四区 | 999精品在线 | 视频一区在线播放 | 久久久久亚洲精品 | 色综合一区二区三区 | 超碰免费人人 | 91香蕉视频在线观看 | 九一亚洲精品 | 亚洲香蕉视频 | 一区二区精品 |