1. 程式人生 > >三個案例帶你看懂LayoutInflater中inflate方法兩個參數和三個參數的區別

三個案例帶你看懂LayoutInflater中inflate方法兩個參數和三個參數的區別

tco rom net roi 異常 com 組成 @override 修改

目錄(?)[+]

關於inflate參數問題,我想很多人多多少少都了解一點,網上也有很多關於這方面介紹的文章,但是枯燥的理論或者翻譯讓很多小夥伴看完之後還是一臉懵逼,so,我今天想通過三個案例來讓小夥伴徹底的搞清楚這個東東。本篇博客我們不講源碼,只看使用。源碼的解讀會在下一篇博文中帶來。

inflate方法從大範圍來看,分兩種,三個參數的構造方法和兩個參數的構造方法。在這兩類中又有細分,OK,那我們就把各種情況都來演示一遍。

1.三個參數的inflate方法

方法頭如下:

[java] view plain copy
  1. public View inflate(@LayoutRes int resource, @Nullable ViewGroup root, boolean attachToRoot)

好,這裏主要分為三種情況,分別來看

1.1 root不為null,attachToRoot為true

當root不為null,attachToRoot為true時,表示將resource指定的布局添加到root中,添加的過程中resource所指定的的布局的根節點的各個屬性都是有效的。比如下面一個案例,我的Activity的布局如下:

[java] view plain copy
  1. <?xml version="1.0" encoding="utf-8"?>
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. xmlns:tools="http://schemas.android.com/tools"
  4. android:layout_width="match_parent"
  5. android:layout_height="match_parent"
  6. android:orientation="vertical"
  7. android:id="@+id/ll"
  8. tools:context="org.sang.layoutinflater.MainActivity">
  9. </LinearLayout>

我還有一個布局linearlayout.xml如下:

[java] view plain copy
  1. <?xml version="1.0" encoding="utf-8"?>
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="200dp"
  4. android:layout_height="200dp"
  5. android:background="@color/colorPrimary"
  6. android:gravity="center"
  7. android:orientation="vertical">
  8. <Button
  9. android:layout_width="wrap_content"
  10. android:layout_height="wrap_content" />
  11. </LinearLayout>

我現在想把這個linearlayout.xml布局文件添加到我的activity的布局中,那麽我可以這麽做:

[java] view plain copy
  1. @Override
  2. protected void onCreate(Bundle savedInstanceState) {
  3. super.onCreate(savedInstanceState);
  4. setContentView(R.layout.activity_main);
  5. LinearLayout ll = (LinearLayout) findViewById(R.id.ll);
  6. LayoutInflater inflater = LayoutInflater.from(this);
  7. inflater.inflate(R.layout.linearlayout, ll,true);
  8. }
小夥伴們註意到,這裏我都沒寫將inflate出來的View添加到ll中的代碼,但是linearlayout布局文件就已經添加進來了,這就是因為我第三個參數設置為了true,表示將第一個參數所指定的布局添加到第二個參數的View中。最終顯示效果如下:

技術分享

如果我作死多寫這麽一行代碼,如下:

[java] view plain copy
  1. protected void onCreate(Bundle savedInstanceState) {
  2. super.onCreate(savedInstanceState);
  3. setContentView(R.layout.activity_main);
  4. LinearLayout ll = (LinearLayout) findViewById(R.id.ll);
  5. LayoutInflater inflater = LayoutInflater.from(this);
  6. View view = inflater.inflate(R.layout.linearlayout, ll, true);
  7. ll.addView(view);
  8. }

這個時候再運行,系統會拋如下異常:

[java] view plain copy
  1. java.lang.IllegalStateException: The specified child already has a parent. You must call removeView() on the child‘s parent first.

原因就是因為當第三個參數為true時,會自動將第一個參數所指定的View添加到第二個參數所指定的View中。

1.2 root不為null,attachToRoot為false

如果root不為null,而attachToRoot為false的話,表示不將第一個參數所指定的View添加到root中,那麽這個時候有的小夥伴可能就有疑問了,既然不添加到root中,那我還寫這麽多幹嘛?我第二個參數直接給null不就可以了?其實不然,這裏涉及到另外一個問題:我們在開發的過程中給控件所指定的layout_width和layout_height到底是什麽意思?該屬性的表示一個控件在容器中的大小,就是說這個控件必須在容器中,這個屬性才有意義,否則無意義。這就意味著如果我直接將linearlayout加載進來而不給它指定一個父布局,則inflate布局的根節點的layout_width和layout_height屬性將會失效(因為這個時候linearlayout將不處於任何容器中,那麽它的根節點的寬高自然會失效)。如果我想讓linearlayout的根節點有效,又不想讓其處於某一個容器中,那我就可以設置root不為null,而attachToRoot為false。這樣,指定root的目的也就很明確了,即root會協助linearlayout的根節點生成布局參數,只有這一個作用。OK,還是上面的布局文件,如果我想將之添加到activity的布局中又該如何呢?

[java] view plain copy
  1. protected void onCreate(Bundle savedInstanceState) {
  2. super.onCreate(savedInstanceState);
  3. setContentView(R.layout.activity_main);
  4. LinearLayout ll = (LinearLayout) findViewById(R.id.ll);
  5. LayoutInflater inflater = LayoutInflater.from(this);
  6. View view = inflater.inflate(R.layout.linearlayout, ll, false);
  7. ll.addView(view);
  8. }

大家註意,這個時候我需要手動的將inflate加載進來的view添加到ll容器中,因為inflate的最後一個參數false表示不將linealayout添加到ll中。顯示效果和上文一樣,不再貼圖。

1.3 root為null

當root為null時,不論attachToRoot為true還是為false,顯示效果都是一樣的。當root為null表示我不需要將第一個參數所指定的布局添加到任何容器中,同時也表示沒有任何容器來來協助第一個參數所指定布局的根節點生成布局參數。我還是使用上文提到的linearlayout,我們來看下面一段代碼:

[java] view plain copy
  1. protected void onCreate(Bundle savedInstanceState) {
  2. super.onCreate(savedInstanceState);
  3. setContentView(R.layout.activity_main);
  4. LinearLayout ll = (LinearLayout) findViewById(R.id.ll);
  5. LayoutInflater inflater = LayoutInflater.from(this);
  6. View view = inflater.inflate(R.layout.linearlayout, null, false);
  7. ll.addView(view);
  8. }

當第二個參數為null,第三個參數為false時(即使為true顯示效果也是一樣的,這裏以false為例),由於在inflate方法中沒有將linearlayout添加到某一個容器中,所以我需要手動添加,另外由於linearlayout並沒有處於某一個容器中,所以它的根節點的寬高屬性會失效,顯示效果如下:

技術分享

小夥伴們註意,這個時候不管我給linearlayout的根節點的寬高設置什麽,都是沒有效果的,它都是包裹button,如果我修改button,則button會立即有變化,因為button是處於某一個容器中的。

2.兩個參數的inflate方法

兩個參數的inflate方法就很簡單了,我們來稍微看一點點源碼: [java] view plain copy
  1. public View inflate(XmlPullParser parser, @Nullable ViewGroup root) {
  2. return inflate(parser, root, root != null);
  3. }

這是兩個參數的inflate方法,大家註意兩個參數實際上最終也是調用了三個參數。 兩個參數的inflate方法分為如下兩種情況: 1.root為null,等同於1.3所述情況。 2.root不為null,等同於1.1所述情況。

3.為什麽Activity布局的根節點的寬高屬性會生效?

inflate方法我們已經說完了,小夥伴們可能有另外一個疑問,那為什麽Activity布局的根節點的寬高屬性會生效?其實原因很簡單,大部分情況下我們一個Activity頁面由兩部分組成(Android的版本號和應用主題會影響到Activity頁面組成,這裏以常見頁面為例),我們的頁面中有一個頂級View叫做DecorView,DecorView中包含一個豎直方向的LinearLayout,LinearLayout由兩部分組成,第一部分是標題欄,第二部分是內容欄,內容欄是一個FrameLayout,我們在Activity中調用setContentView就是將View添加到這個FrameLayout中,所以給大家一種錯覺仿佛Activity的根布局很特殊,其實不然。

OK,以上就是對LayoutInflater中inflate方法的一個簡單介紹,希望能夠幫助到還沒弄懂這個的小夥伴。

以上。

三個案例帶你看懂LayoutInflater中inflate方法兩個參數和三個參數的區別