简介:
ViewStub是一个不可见的,能在运行期间延迟加载的大小为0的View,它直接继承于View。当对一个ViewStub调用 inflate()方法或设置它可见时,系统会加载在ViewStub标签中引入的我们自己定义的View,然后填充在父布局当中。也就是说,在对 ViewStub调用inflate()方法或设置visible之前,它是不占用布局空间和系统资源的。它的使用场景可以是在我们需要加载并显示一些不常用的View时,ViewStub会一直正在View视图层,直到调用了setVisibility(int)或者inflate()方法。
注意点
1.当布局文件inflate时,ViewStub控件虽然也占据内存,但是相相比于其他控件,ViewStub所占内存很小;
2.布局文件inflate时,ViewStub主要是作为一个“占位符”的性质,放置于view tree中,且ViewStub本身是不可见的。ViewStub中有一个layout属性,指向ViewStub本身可能被替换掉的布局文件,在一定时机时,通过viewStub.inflate()或者setVisibility(int);
3.ViewStub本身是不可见的所以在布局文件按里面设置可见性是没有任何效果的,对ViewStub setVisibility(..)与其他控件不一样,ViewStub的setVisibility 成View.VISIBLE或INVISIBLE如果是首次使用,都会自动inflate其指向的布局文件。
setVisibility会在内部调用inflate()方法。
@Override
@android.view.RemotableViewMethod
public void setVisibility(int visibility) {
if (mInflatedViewRef != null) {
View view = mInflatedViewRef.get();
if (view != null) {
view.setVisibility(visibility);
} else {
throw new IllegalStateException("setVisibility called on un-referenced view");
}
} else {
super.setVisibility(visibility);
if (visibility == VISIBLE || visibility == INVISIBLE) {
inflate();
}
}
}
4中所讲到的ViewStub指向的布局文件解析inflate并替换掉当前ViewStub本身,并不是完全意义上的替换(与include标签还不太一样),替换时,布局文件的layout params是以ViewStub为准,其他布局属性是以布局文件自身为准。
5.可以通过isInflated()方法,判断ViewStub是够已经inflate过布局文件,避免再次inflate造成的程序错误。
6ViewStub只能Inflate一次,之后ViewStub对象会被置为空。按句话说,某个被ViewStub指定的布局被Inflate后,就不会够再通过ViewStub来控制它了。
代码实现:
1.布局文件:
<?xml version="1.0" encoding="utf-8"?>
<layout
xmlns:android="http://schemas.android.com/apk/res/android"
>
<FrameLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
>
<ViewStub
android:visibility="visible"
android:id="@+id/vsGuide"
android:layout_width="match_parent"
android:layout_height="190.0dp"
android:layout="@layout/include_guide">
</ViewStub>
<ViewStub
android:id="@+id/vsWelcome"
android:layout_width="match_parent"
android:layout_height="190.0dp"
android:layout="@layout/include_welcome">
</ViewStub>
<Button
android:text="Inflate the ViewStub"
android:onClick="inflateViewStub"
android:layout_width="wrap_content"
android:layout_height="wrap_content" />
</FrameLayout>
</layout>
activity的实现:
通过按钮时间控制inflate的出发,这里面用到了Data Binding。会出现红色的错误提示,可以忽略。
public void inflateViewStub(View view)
{
if(!binding.vsGuide.isInflated())
{
binding.vsGuide.getViewStub().setVisibility(View.GONE);
}
}