小黑与小白的故事,通过虚拟这两个人物进行一问一答的形式来共同学习ViewStub的使用

小白:Hi,小黑,ViewStub是什么?听说能够用来进行布局优化。

小黑:ViewStub 是一个隐藏的,不占用内存空间的视图对象。它能够在执行时延迟载入布局资源文件。(很多其它具体的API等信息能够查看官方文档ViewStub),计算机行业一向是实践里面出真知,以下用一个样例演示下效果。


小黑:说说概念仅仅是为了概括性的了解下。还是用个实例来演示下。先来创建一个Activity中使用的布局文件,文件名称是: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="horizontal" > <Button
android:id="@+id/show_button"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="显示"/> <ViewStub
android:id="@+id/viewstub"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout="@layout/sub_layout"
/> <Button
android:id="@+id/hide_button"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="隐藏"/> </LinearLayout>

小白:“显示”、“隐藏”字符串没有放入values/string.xml。控件的id也没有一定的命名规则的乱起。


小黑:。。。。。。“你是猴子请来的救兵吗?”。一切从简好吧。注意上面的ViewStub的使用方法。当中android:layout="@layout/sub_layout"引入一个新的布局。sub_layout.xml代码例如以下:
<TextView  xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/textview"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="ViewStub中包括的TextVeiw"/>

小白:如今有两个布局文件了,一个是Activity setContentView须要的activity_main.xml,一个是当中引入的sub_layout

小黑:以下是一个MainActivity.java。加入些点击事件
package com.example.viewstub;

import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.view.ViewStub;
import android.view.View.OnClickListener; public class MainActivity extends Activity { @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main); final ViewStub viewStub = (ViewStub) findViewById(R.id.viewstub); View showButton = findViewById(R.id.show_button);
showButton.setOnClickListener(new OnClickListener() { @Override
public void onClick(View v) {
viewStub.inflate();
}
}); View hideButton = findViewById(R.id.hide_button);
hideButton.setOnClickListener(new OnClickListener() { @Override
public void onClick(View v) {
viewStub.setVisibility(View.GONE);
}
});
}
}

小黑:以下是代码执行后,默认并没有点击显示button的情况下。使用“DDMS -> Dump View Hierarchy for UI Automator”工具的截图



小白:我发现activity_main.xml布局文件在“显示”,“隐藏”两个button之间有一个<ViewStub>而在上面的截图中并没有。说明ViewStub在初始化载入时并不会加入到视图树上(Android UI Tree)。

我也使用“Hierarchy View”工具的截图例如以下:






小黑:对,这是ViewStub能够进行布局优化的地方“懒载入视图”,初始化时系统不会初始化ViewStub引用的视图。

再来看下接着看下点击“显示”button后的截图。





小黑:ViewStub使用流程是 1. 布局中加入ViewStub (XML加入、代码中加入) 2. inflate显示 3. setVisibility隐藏。


小白:通过在代码中viewStub.inflate(); ViewStub引用的布局显示出来了。只是,这不就是动态的加入视图吗?与View.setVisibility(View.GONE);有啥差别?

小黑:ViewStub是一个没有尺寸大小而且不会在布局中嵌套或渲染不论什么东西的轻量级的视图。因此在视图层次展现或隐藏它的代价很小。当ViewStub可见,或者调用 inflate()函数时,才会载入这个布局资源文件。

该ViewStub在载入视图时在父容器中替换它本身。

因此,ViewStub会一直存在于视图中。直到调用setVisibility(int) 或者inflate()为止。ViewStub的布局參数会随着载入的视图数一同被加入到ViewStub父容器。相同,你也能够通过使用inflatedId属性来定义或重命名要载入的视图对象的Id值。

    View.setVisibility(View.GONE); 方式在初始化时就会加入到视图树上(Android UI Tree),而使用ViewStub在初始化时不会加入。

小白:使用ViewStub有啥须要注意的吗?
小黑:1. 在要渲染的布局中并不支持<merge/>标签。

2. ViewStub.infalte方法不能调用两次。否者会出现下面异常:


java.lang.IllegalStateException: ViewStub must have a non-null ViewGroup viewParent
at android.view.ViewStub.inflate(ViewStub.java:287)
at com.example.viewstub.MainActivity$1.onClick(MainActivity.java:23)
at android.view.View.performClick(View.java:4475)
at android.view.View$PerformClick.run(View.java:18786)
at android.os.Handler.handleCallback(Handler.java:730)
at android.os.Handler.dispatchMessage(Handler.java:92)
at android.os.Looper.loop(Looper.java:176)
at android.app.ActivityThread.main(ActivityThread.java:5419)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:525)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1209)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1025)
at dalvik.system.NativeStart.main(Native Method)

小白:在网上看的资料说是ViewStub的一个缺点是引入的布局不能使单独的视图,而必须是一个布局才行,是这样吗?
小黑:这样的观点是错误的。在样例sub_layout.xml 中仅有一个TextView视图。

小白:为什么ViewStub能够作为一个布局标签使用?
小黑:你能够查看一下ViewStub的源代码。public
final class ViewStub extends View。 ViewStub本身就是一个View 子类。


小白:我有几个问题:
1. ViewStub加入的视图能够动态删除吗?
2. 不能第二次inflate为什么?
3. 尽管第二次不能直接inflate,能够直接删除,代码中加入视图吗?
4. 除了调用inflate方法还能通过其它方式显示出来吗?(找到id 设置visible)

小黑:偶不知道啊。



补充:ViewStub显示有两种办法。上面介绍的是使用inflate方法,还能够直接在ViewStub.setVisibiltity(View.Visible)

异常情况:
    <ViewStub
android:id= "@+id/view_stub_text"
android:layout_width= "wrap_content"
android:layout_height="wrap_content"
>
<TextView
android:layout_width= "wrap_content"
android:layout_height="wrap_content"
android:text="love_world_"
/>
</ViewStub >

出现下面异常:

java.lang.ClassCastException: android.view.ViewStub cannot be cast to android.view.ViewGroup

使用场景:
ListView ItemView
     个人评估是否有必要
     Gone是否也已经优化?


參考资料
ViewStub - 官方文档
Loading Views On Demand - 官方教程


很多其它优化相关的文章详见:《Android 基础学习文章汇总》 第三部分 性能优化


2014-11-12  加入异常与使用场景

Android 性能优化 三 布局优化ViewStub标签的使用的更多相关文章

  1. Android性能优化:布局优化 详细解析(含<include>、<ViewStub>、<merge>讲解 )

    1. 影响的性能 布局性能的好坏 主要影响 :Android应用中的页面显示速度 2. 如何影响性能 布局影响Android性能的实质:页面的测量 & 绘制时间 1个页面通过递归 完成测量 & ...

  2. Android性能优化之布局优化

    最新最准确内容建议直接访问原文:Android性能优化之布局优化 本文为Android性能优化的第二篇——布局优化,主要介绍使用抽象布局标签(include, viewstub, merge).去除不 ...

  3. 【转】Android性能优化之布局优化篇

     转自:http://blog.csdn.net/feiduclear_up/article/details/46670433 Android性能优化之布局优化篇 分类: andorid 开发2015 ...

  4. Android 性能优化 四 布局优化merge标签的使用

    小白:之前分享了ViewStub标签的使用,Android还有其他优化布局的方式吗? 小黑:<merge />标签用于减少View树的层次来优化Android的布局.先来用个例子演示一下: ...

  5. Android Studido下的应用性能优化总结--布局优化

    前言:一个应用的成功=产品设计*性能 ,再此我们不讨论一个应用的设计,那交给我们可爱又可恨的产品经理和UI设计师来决定!所以这里分步骤讨论如何提升一个应用的性能,这里先探讨布局优化问题. 布局优化 避 ...

  6. Android开发学习之路--性能优化之布局优化

      Android性能优化方面也有很多文章了,这里就做一个总结,从原理到方法,工具等做一个简单的了解,从而可以慢慢地改变编码风格,从而提高性能. 一.Android系统是如何处理UI组件的更新操作的 ...

  7. Android最佳性能实践(四)——布局优化技巧

    转载请注明出处:http://blog.csdn.net/guolin_blog/article/details/43376527 在前面几篇文章其中.我们学习了怎样通过合理管理内存,以及高性能编码技 ...

  8. android 优化之布局优化

    布局优化的思路很简单,尽量减少布局文件的层级,看过系统源码的都知道,Android view绘制都是逐层绘制的,所以布局的层级少了,decodeview的时候绘制工作自然就少了. 那么如何进行布局的优 ...

  9. Android 卡顿优化 4 布局优化实际技巧

    今天分享一些layout布局书写中的一些技巧,希望看过之后你也一样可以写出性价比高的布局.我个人的目标是用最少的View写出一样效果的布局.因为我相信View的数量减少伴随着的就是层级的减少.从而达到 ...

随机推荐

  1. BZOJ 1415: [Noi2005]聪聪和可可( 最短路 + 期望dp )

    用最短路暴力搞出s(i, j)表示聪聪在i, 可可在j处时聪聪会走的路线. 然后就可以dp了, dp(i, j) = [ dp(s(s(i,j), j), j) + Σdp(s(s(i,j), j), ...

  2. 通过netty实现服务端与客户端的长连接通讯,及心跳检测。

    基本思路:netty服务端通过一个Map保存所有连接上来的客户端SocketChannel,客户端的Id作为Map的key.每次服务器端如果要向某个客户端发送消息,只需根据ClientId取出对应的S ...

  3. win7中注册tomcat服务

    非安装版tomcat下载后,在bin文件夹会有一个startup.bat文件,运行该文件即可启动tomcat了.不过在服务器配置tomcat的话,就通常需要注册为服务. 在/bin文件下还有tomca ...

  4. Date对象需要注意的点

    var today=new Date(); Date对象取得了PC内部时钟的一个快照,并同时返回一个Date对象实例. 注意静态Date对象和Date对象实例的差别,后者包含一个实际的日期值.毫秒为单 ...

  5. 再探Delphi2010 Class的构造和析构顺序

    发了上一篇博客.盒子上有朋友认为Class的构造和析构延迟加载.是在Unit的初始化后调用的Class的构造.在Unit的反初始化前调用的Class的析构函数. 为了证明一下我又做了个试验 unit ...

  6. ComboBox控件绑定数据源

    最近在研究机房收费系统的组合查询的方法时,看到了ComboBox控件可以进行数据绑定,我觉得这个功能真的很不错,可以给我省去很多的麻烦. 下面是我组合查询窗体界面 一.数据转换方法 现在我们开看一下我 ...

  7. tar.xz文件怎样解压

    XZ压缩最新压缩率之王 xz这个压缩可能非常多都非常陌生,只是您可知道xz是绝大数linux默认就带的一个压缩工具. 之前xz使用一直非常少,所以差点儿没有什么提起. 我是在下载phpmyadmin的 ...

  8. CVT电子集团--笔试部分试题

    之前有在网上答了下CVT的网上笔试题,特别把它们都弄下来,答案参考,不一定是对的,有错希望大家能提出来. 1.有关系R和S,R∩S等价于(B) A.S-(R-S)    B.R-(R-S)   C.( ...

  9. 传统web和mvc的区别

  10. 笔记 postgresql oid同步

    以前学习postgresql的笔记 create table 消耗 OID 如create table my_test_table, 他本身会消耗一个 会在pg_type中插入两条记录_my_test ...