溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

如何在Android中使用LayoutInflater.inflater()

發(fā)布時(shí)間:2021-06-08 16:58:21 來源:億速云 閱讀:143 作者:Leah 欄目:移動(dòng)開發(fā)

如何在Android中使用LayoutInflater.inflater()?相信很多沒有經(jīng)驗(yàn)的人對(duì)此束手無策,為此本文總結(jié)了問題出現(xiàn)的原因和解決方法,通過這篇文章希望你能解決這個(gè)問題。

首先來看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;
}

其實(shí)就是從Context中獲取Context.LAYOUT_INFLATER_SERVICE所對(duì)應(yīng)的系統(tǒng)服務(wù)。這里涉及到Context實(shí)現(xiàn)以及服務(wù)創(chuàng)建的源碼,不繼續(xù)深究。

重點(diǎn)是通常所使用的inflate()方法,比較常用的就是這兩個(gè):

  • inflate(@LayoutRes int resource, @Nullable ViewGroup root)

  • inflate(@LayoutRes int resource, @Nullable ViewGroup root, boolean attachToRoot)

另外兩個(gè)方法inflate(XmlPullParser parser, @Nullable ViewGroup root)inflate(XmlPullParser parser, @Nullable ViewGroup root, boolean attachToRoot)

而兩個(gè)參數(shù)的方法,實(shí)際也是調(diào)用了三個(gè)參數(shù)的inflate()方法,只是在三個(gè)參數(shù)傳入了root!=null

public View inflate(@LayoutRes int resource, @Nullable ViewGroup root) {
 return inflate(resource, root, root != null);
}

那我們就可以直接看三個(gè)參數(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();
 }
}

這里其實(shí)就會(huì)發(fā)現(xiàn),最后return調(diào)用的其實(shí)是inflate(XmlPullParser parser, @Nullable ViewGroup root, boolean attachToRoot)這個(gè)方法,所謂的四個(gè)inflate()方法,其他三個(gè)只是對(duì)這個(gè)方法的重載,主要代碼還是在這個(gè)方法中實(shí)現(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對(duì)象轉(zhuǎn)換成AttributeSet
 final AttributeSet attrs = Xml.asAttributeSet(parser);
 Context lastContext = (Context) mConstructorArgs[0];
 mConstructorArgs[0] = inflaterContext;
 View result = root;

 try {
  //2.在xml中尋找根節(jié)點(diǎn),如果類型是XmlPullParser.START_TAG或者XmlPullParser.END_DOCUMENT就會(huì)退出循環(huán)
  int type;
  while ((type = parser.next()) != XmlPullParser.START_TAG &&
   type != XmlPullParser.END_DOCUMENT) {
  // Empty
  }
  //3.如果根節(jié)點(diǎn)類型不是XmlPullParser.START_TAG將拋出異常
  if (type != XmlPullParser.START_TAG) {
  throw new InflateException(parser.getPositionDescription()
   + ": No start tag found!");
  }

  final String name = parser.getName();

  //4.判斷根節(jié)點(diǎn)是否是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é)點(diǎn)創(chuàng)建臨時(shí)的view對(duì)象
  final View temp = createViewFromTag(root, name, inflaterContext, attrs);

  ViewGroup.LayoutParams params = null;

  if (root != null) {
   //6.如果root不為空,則調(diào)用generateLayoutParams(attrs)獲取root所對(duì)應(yīng)LayoutParams對(duì)象
   params = root.generateLayoutParams(attrs);
   //是否attachToRoot
   if (!attachToRoot) {
   //7.如果attachToRoot為false,則使用root默認(rèn)的LayoutParams作為臨時(shí)view對(duì)象的屬性
   temp.setLayoutParams(params);
   }
  }

  //8.inflate xml的所有子節(jié)點(diǎn)
  rInflateChildren(parser, temp, attrs, true);

  //9.判斷是否需要將創(chuàng)建的臨時(shí)view attach到root中
  if (root != null && attachToRoot) {
   root.addView(temp, params);
  }

  //10.決定方法的返回值是root還是臨時(shí)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則是很多時(shí)候使用inflate方法之后,發(fā)現(xiàn)xml布局設(shè)置的寬高屬性不生效的部分原因,有時(shí)候在RecyclerView中添加就會(huì)這樣。如果root!=null且attachToRoot為false時(shí),創(chuàng)建的view則會(huì)具有自身根節(jié)點(diǎn)屬性值,與root對(duì)應(yīng)的LayoutParam

9的判斷決定了創(chuàng)建的view是否添加到root中,而10則決定了方法返回的是root還是view

總結(jié)

根據(jù)inflate的參數(shù)不同可以獲得不同的返回值

rootattachToRoot返回值
nullfalse(或者true)返回resource對(duì)應(yīng)的view對(duì)象,但是xml中根節(jié)點(diǎn)的屬性沒有生效
!=nullfalse返回resource對(duì)應(yīng)的view對(duì)象,并且xml中根節(jié)點(diǎn)的屬性生效,view對(duì)象的LayoutParam與root的LayoutParam對(duì)應(yīng)
!=nulltrue返回root對(duì)象,對(duì)應(yīng)resource創(chuàng)建的view對(duì)象,xml中根節(jié)點(diǎn)的屬性生效,并且將會(huì)添加到root中

注意:attachToRoot默認(rèn)為root!=null的值

看完上述內(nèi)容,你們掌握如何在Android中使用LayoutInflater.inflater()的方法了嗎?如果還想學(xué)到更多技能或想了解更多相關(guān)內(nèi)容,歡迎關(guān)注億速云行業(yè)資訊頻道,感謝各位的閱讀!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI