中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

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

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

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

發布時間:2021-06-08 16:58:21 來源:億速云 閱讀:149 作者:Leah 欄目:移動開發

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

首先來看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所對應的系統服務。這里涉及到Context實現以及服務創建的源碼,不繼續深究。

重點是通常所使用的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)

而兩個參數的方法,實際也是調用了三個參數的inflate()方法,只是在三個參數傳入了root!=null

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

那我們就可以直接看三個參數的inflate()方法了,其中res.getLayout(resource)這句代碼,已經將我們傳入的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();
 }
}

這里其實就會發現,最后return調用的其實是inflate(XmlPullParser parser, @Nullable ViewGroup root, boolean attachToRoot)這個方法,所謂的四個inflate()方法,其他三個只是對這個方法的重載,主要代碼還是在這個方法中實現的

這部分代碼較長,以注釋的形式解釋代碼

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中尋找根節點,如果類型是XmlPullParser.START_TAG或者XmlPullParser.END_DOCUMENT就會退出循環
  int type;
  while ((type = parser.next()) != XmlPullParser.START_TAG &&
   type != XmlPullParser.END_DOCUMENT) {
  // Empty
  }
  //3.如果根節點類型不是XmlPullParser.START_TAG將拋出異常
  if (type != XmlPullParser.START_TAG) {
  throw new InflateException(parser.getPositionDescription()
   + ": No start tag found!");
  }

  final String name = parser.getName();

  //4.判斷根節點是否是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.通過根節點創建臨時的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的所有子節點
  rInflateChildren(parser, temp, attrs, true);

  //9.判斷是否需要將創建的臨時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中跟布局的屬性數據。

6,7則是很多時候使用inflate方法之后,發現xml布局設置的寬高屬性不生效的部分原因,有時候在RecyclerView中添加就會這樣。如果root!=null且attachToRoot為false時,創建的view則會具有自身根節點屬性值,與root對應的LayoutParam

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

總結

根據inflate的參數不同可以獲得不同的返回值

rootattachToRoot返回值
nullfalse(或者true)返回resource對應的view對象,但是xml中根節點的屬性沒有生效
!=nullfalse返回resource對應的view對象,并且xml中根節點的屬性生效,view對象的LayoutParam與root的LayoutParam對應
!=nulltrue返回root對象,對應resource創建的view對象,xml中根節點的屬性生效,并且將會添加到root中

注意:attachToRoot默認為root!=null的值

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

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

建始县| 分宜县| 时尚| 杭锦后旗| 汨罗市| 秭归县| 太和县| 千阳县| 玉山县| 酒泉市| 石楼县| 岱山县| 永川市| 毕节市| 澎湖县| 台东市| 电白县| 西宁市| 额敏县| 徐水县| 历史| 若尔盖县| 延边| 英超| 盐池县| 石屏县| 西林县| 阳江市| 鄂托克前旗| 九寨沟县| 德昌县| 雷山县| 会同县| 临江市| 伊宁市| 河南省| 航空| 庐江县| 白玉县| 清丰县| 黄陵县|