1. 程式人生 > >Android中的viewStub、requestFocus、include、merge標籤

Android中的viewStub、requestFocus、include、merge標籤

<ViewStub/>標籤

ViewStub是一個不可見的,大小為0的View,最佳的用途就是實現View的延遲載入,在需要的時候再載入View。當呼叫ViewStub的setVisibility方法設定為可見或者呼叫inflate()方法初始化該View的時候,ViewStub引用的資源開始初始化,然後引用的資源會替代掉ViewStub,把自己填充在ViewStub的原位置。因此在沒有呼叫setVisibility(int)或inflate()方法之前ViewStub會一直存在元件樹層級結構中,但是由於ViewStub非常輕量級,這對效能影響非常小。

例如下面的例子:

view_stub_btn.xml

<?xml version="1.0" encoding="utf-8"?>
<Button xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/viewStubBtn"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:text="ViewStubBtn" />

main.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" >

    <View
        android:layout_width="match_parent"
        android:layout_height="0dip"
        android:layout_weight="1" />

    <Button
        android:id="@+id/showViewStubBtn"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:gravity="center"
        android:text="showViewStubView" />

    <ViewStub
        android:id="@+id/viewStub"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout="@layout/view_stub_btn" />


</LinearLayout>
MainActivity.java

package com.ygc;

import com.example.androidtest.R;

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

public class MainActivity extends Activity {
    private ViewStub mViewStub;
    private Button mShowViewStubBtn;
    private Button mViewStubBtn;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        mViewStub = (ViewStub) findViewById(R.id.viewStub);
        mShowViewStubBtn = (Button) findViewById(R.id.showViewStubBtn);
        mShowViewStubBtn.setOnClickListener(new OnClickListener() {

            @Override
            public void onClick(View v) {
                if (mViewStubBtn == null) {
                    mViewStubBtn = (Button) mViewStub.inflate();
                    mViewStubBtn.setOnClickListener(new OnClickListener() {

                        @Override
                        public void onClick(View v) {
                            mViewStubBtn.setVisibility(View.GONE);
                        }
                    });
                } else {
                    mViewStubBtn.setVisibility(View.VISIBLE);
                }
            }
        });

    }

}

在沒有載入ViewStub時顯示的效果如下:


通過Hierarchy Viewer工具檢視View的層次結構如下:


展開ViewStub可以得到載入它所用到的時間:


可以看到,所有過程用到的時間都是0ms。

載入ViewStub後的效果為:


通過Hierarchy Viewer工具檢視載入完ViewStub後的View的層次結構如下:


展開載入完的Button得到載入過程所需的時間為:


可以看到載入的每個過程所花費的時間。

當呼叫inflate()方法的時候,ViewStub被引用的資源替代,並且返回引用的View。這樣程式可以直接得到引用的View而不用再次呼叫findViewById()方法來查找了。只能在一個ViewStub物件上呼叫一次inflate()方法,因為當呼叫inflate()方法後原來的ViewStub物件已經被引用的資源所代替,ViewStub會從父元件中刪除,所以當你再次在同一個ViewStub物件上呼叫inflate()方法時會得到java.lang.IllegalStateException: ViewStub must have a non-null ViewGroup viewParent。


注:(1) ViewStub目前有個缺陷就是還不支援 <merge /> 標籤。

          (2)<ViewStub/>標籤需要指定layout_width和layout_height屬性,它們會覆蓋被載入的layout中的這些屬性。

如果你在ViewStub標籤中包含的是一個帶有merge標籤的layout,如下:

<?xml version="1.0" encoding="utf-8"?>
<merge xmlns:android="http://schemas.android.com/apk/res/android" >

    <Button
        android:id="@+id/viewStubBtn"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="ViewStubBtn" />

</merge>
在載入的時候會得到 android.view.InflateException: <merge /> can be used only with a valid ViewGroup root and attachToRoot=true。