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" />
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();
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 >
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 >
< 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 ;
}
}
}
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 ;
}
}
}
转自:http://www.blogjava.net/gaolei-xj/archive/2013/02/17/395348.html