<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。