在4.5.6節介紹過一個<include>標籤,該標籤可以在布局檔案中引用另外一個布局檔案,並可以覆蓋被引用布局檔案根節點所有與布局相關的屬性,也就是以android:layout開頭的屬性。通過<include>標籤可以將一個非常龐大的布局檔案分解成若干個較小的布局檔案,而且這些小的布局檔案也可以被多次引用,從而達到一個重用的目的。
<include>標籤固然很好用,但有一個問題,就是布局檔案中的控制項並不一定在程式啟動時全都用到,有一些控制項只在特定的情況下才會被使用到。例如,一個閱讀圖書的軟體只有在下載電子書時才需要顯示進度條,在平時看書時都是裝載的本地電子書,並不需要使用進度條。因此,在程式啟動時完全可以先不載入這個進度條。但使用<include>標籤引用這個包含進度條的布局檔案時,不管三七二十一,將所有的控制項全部裝載到了記憶體中。也許有的讀者會說,一個進度條佔用不了多少系統資源,都裝載也無所謂。這些讀者也許是對的,但如果裝載的不是進度條,而是很多ImageView控制項(顯示了很大的映像),並且還不是在一個地方裝載,那恐怕就會將可憐的手機資源消耗殆盡了。因此,我們急需一種機制來改變<include>標籤的這種行為,只在需要時裝載控制項。這種機制就是本節要介紹的ViewStub控制項。
ViewStub是不可視的控制項,它的作用與<include>標籤基本相同,在布局檔案中使用<ViewStub>標籤來引用其他的布局檔案。但與<include>唯一的不同是ViewStub並不會馬上裝載引用的布局檔案。只有在調用了ViewStub.inflate或ViewStub.setVisibility(View.VISIBLE)方法後,ViewStub才會裝載引用的控制項,下面先看兩個布局檔案。
main.xml 複製代碼 代碼如下:<?xmlversion="1.0"encoding="utf-8"?>
<LinearLayoutxmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"android:layout_width="fill_parent"
android:layout_height="fill_parent">
<Buttonandroid:layout_width="fill_parent"
android:layout_height="wrap_content"android:text="我的按鈕"
android:onClick="onClick_Button"/>
<includelayout="@layout/custom"/>
</LinearLayout>
custom.xml 複製代碼 代碼如下:<?xmlversion="1.0"encoding="utf-8"?>
<LinearLayoutxmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"android:layout_width="fill_parent"
android:layout_height="fill_parent">
<Buttonandroid:layout_width="fill_parent"
android:layout_height="wrap_content"android:text="按鈕1"/>
<Buttonandroid:layout_width="fill_parent"
android:layout_height="wrap_content"android:text="按鈕2"/>
</LinearLayout>
在main.xml檔案中使用了<include>標籤來引用custom.xml,在這種情況下,螢幕上會立即顯示三個5.56所示的按鈕。如果將<include>標籤換成如下的代碼,在程式啟動時,只會顯示在main.xml檔案中的定義按鈕,5.57所示。 複製代碼 代碼如下:<ViewStubandroid:id="@+id/viewstub"android:inflatedId="@+id/button_layout"
android:layout="@layout/custom"android:layout_width="fill_parent"
android:layout_height="wrap_content"/>
圖5.56 使用<include>標籤裝載控制項
圖5.56使用<ViewStub>標籤裝載控制項
在使用<ViewStub>標籤引用布局檔案後,還需要調用ViewStub.inflate或ViewStub.setVisibility(View.VISIBLE)方法才能裝載所引用的控制項,代碼如下: 複製代碼 代碼如下:publicvoidonClick_Button(Viewv)
{
//ViesStub控制項只能獲得一次,第二次再使用findViewById獲得該ViewStub對象,則返回null
Viewview=findViewById(R.id.viewstub);
if(view!=null)
{
//或調用ViewStub.inflate方法
//view=((ViewStub)view).inflate();
//裝載ViewStub引用的custom.xml檔案中的控制項
((ViewStub)view).setVisibility(View.VISIBLE);
}
else
{
setTitle("viewisnull");
}
}
單擊“我的按鈕”後,會顯示在custom.xml檔案中定義的兩個按鈕,效果與圖5.56完全一樣。
注意:<ViewStub>與<include>標籤一樣,也可以設定所引用布局檔案中根節點所有與布局相關的屬性。所不同的是<include>標籤的android:id屬性直接覆蓋了所引用布局檔案中根節點的android:id屬性值,而<ViewStub>標籤的android:id屬性與普通控制項標籤的android:id屬性一樣,用於在代碼中引用控制項。在<ViewStub>標籤中需要使用android:inflatedId屬性覆蓋所引用布局檔案中根節點的android:id屬性值。雖然<ViewStub>可完全取代<include>,但唯一的不足是<ViewStub>目前還無法取代<merge>。