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
MainActivity.java<?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>
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。