ViewStub 是一個不可見的,大小為0的View,最佳用途就是實現View的延遲加載,在需要的時候再加載View,可Java中常見的性能優化方法延遲加載一樣。
     當調用ViewStub的setVisibility函數設置為可見或則調用 inflate初始化該View的時候,ViewStub引用的資源開始初始化,然后引用的資源替代ViewStub自己的位置填充在ViewStub的 位置。因此在沒有調用setVisibility(int) 或則 inflate()函數之前 ViewStub一種存在組件樹層級結構中,但是由于ViewStub非常輕量級,這對性能影響非常小。 可以通過ViewStub的inflatedId屬性來重新定義引用的layout id。 例如:
    
<ViewStub android:id="@+id/stub"
               android:inflatedId
="@+id/subTree"
               android:layout
="@layout/mySubTree"
               android:layout_width
="120dip"
               android:layout_height
="40dip" />
上面定義的ViewStub ,可以通過id “stub”來找到,在初始化資源“mySubTree”后,stub從父組件中刪除,然后"mySubTree"替代stub的位置。初始資源"mySubTree"得到的組件可以通過inflatedId 指定的id "subTree"引用。 然后初始化后的資源被填充到一個120dip寬、40dip高的地方。

推薦使用下面的方式來初始化ViewStub:
ViewStub stub = (ViewStub) findViewById(R.id.stub);
     View inflated = stub.inflate();


當調用inflate()函數的時候,ViewStub 被引用的資源替代,并且返回引用的view。 這樣程序可以直接得到引用的view而不用再次調用函數 findViewById()來查找了。


activity_main.xml:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools
="http://schemas.android.com/tools"
    android:layout_width
="match_parent"
    android:layout_height
="match_parent"
    android:orientation
="vertical"
    android:paddingBottom
="@dimen/activity_vertical_margin"
    android:paddingLeft
="@dimen/activity_horizontal_margin"
    android:paddingRight
="@dimen/activity_horizontal_margin"
    android:paddingTop
="@dimen/activity_vertical_margin"
    tools:context
=".MainActivity" >

    
<!-- 靜態加載布局文件 -->

    
<include
        
android:layout_width="wrap_content"
        android:layout_height
="wrap_content"
        layout
="@layout/my_sub_tree" />

    
<!-- 動態加載布局文件 -->

    
<ViewStub
        
android:id="@+id/stub"
        android:layout_width
="wrap_content"
        android:layout_height
="wrap_content"
        android:inflatedId
="@+id/subTree"
        android:layout
="@layout/my_sub_tree" />

    
<Button
        
android:id="@+id/button_show"
        android:layout_width
="wrap_content"
        android:layout_height
="wrap_content"
        android:text
="動態加載布局" />

    
<Button
        
android:id="@+id/button_hidden"
        android:layout_width
="wrap_content"
        android:layout_height
="wrap_content"
        android:text
="動態隱藏布局" />

</LinearLayout>


my_sub_tree.xml:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width
="match_parent"
    android:layout_height
="match_parent"
    android:orientation
="vertical" >

    
<RatingBar
        
android:id="@+id/ratingBar"
        android:layout_width
="wrap_content"
        android:layout_height
="wrap_content" />

</LinearLayout>

MainActivity.java:
package com.example.android_viewstub;

import android.app.Activity;
import android.os.Bundle;
import android.view.Menu;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.ViewStub;
import android.widget.Button;
import android.widget.RatingBar;

/**
 * Demo描述: 利用ViewStub顯示和隱藏布局 ViewStub的引入: 在開發的時候,有些布局是要根據條件而動態顯示,達到一個布局兩用的效果,
 * 運用View.VISIBLE和View.GONE去改變布局的可見性. 這樣的做法顯然是沒什么多大的問題,優點邏輯清晰,控制靈活,但缺點就是耗費資源
 * 在setContentView()或者用inflate加載布局文件時無論View是否
 * 被設置為View.GONE和View.VISIBLE,都會創建對象,占用一定程度上的內存,所以在考慮優化程序的時候,
 * 盡量避免資源浪費,降低程序的資源占有量,提高響應速度,提升軟件的用戶體驗
 * 
 * 推薦的做法是使用android.view.ViewStub. ViewStub是一個輕量級的View,它一個看不見的,不占布局位置,占用資源非常小的控件.
 * ViewStub是一個隱藏的,不占用內存空間的視圖對象,它可以在運行時延遲加載布局資源文件當 ViewStub可見,或者調用
 * inflate()函數時,才會加載這個布局資源文件 注意的問題: ViewStub只能用來Inflate一個布局文件,而不是某個具體的View
 * 
 * 遇到的問題: 報錯 ViewStub must have a non-null ViewGroup viewParent 原因:
 * 官方文檔:viewstub不能反復inflate,只能inflate一次
 *

 
*/
public class MainActivity extends Activity implements OnClickListener {
    
private ViewStub mViewStub;
    
private Button mShowButton;
    
private Button mHiddenButton;

    @Override
    
protected void onCreate(Bundle savedInstanceState) {
        
super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        findView();
        bindView();
    }

    
private void findView() {
        mViewStub 
= (ViewStub) this.findViewById(R.id.stub);
        mShowButton 
= (Button) this.findViewById(R.id.button_show);
        mHiddenButton 
= (Button) this.findViewById(R.id.button_hidden);
    }

    
private void bindView() {
        mShowButton.setOnClickListener(
this);
        mHiddenButton.setOnClickListener(
this);
    }

    @Override
    
public boolean onCreateOptionsMenu(Menu menu) {
        getMenuInflater().inflate(R.menu.main, menu);
        
return true;
    }

    @Override
    
public void onClick(View v) {
        
switch (v.getId()) {
        
case R.id.button_show:
            View inflated 
= mViewStub.inflate();
            RatingBar ratingBar 
= (RatingBar) inflated
                    .findViewById(R.id.ratingBar);
            ratingBar.setRating(
4);
            
// mViewStub.setVisibility(View.VISIBLE);
            break;
        
case R.id.button_hidden:
            mViewStub.setVisibility(View.GONE);
            
break;
        
default:
            
break;
        }
    }

}