Android中所有的界面绘制工作都是在UI线程中进行的,提高UI流畅度的最核心根本在于释放UI线程。即:不在主线程中做耗时的操作。

很多人都知道,耗时的操作要放到子线程中去做,比如访问网络,比如读写sd卡。像这类操作大家都会很自然的想到使用子线程来完成耗时的操作,等操作结束之后,再通过Handler通知主线程进行界面的更新。这是非常正确的方法。但是有一类方法,它必须得运行在在UI线程中,就是布局文件的加载。如果这类方法花的时间太多了,也是会对流畅度产生很大的影响。今天我们就来讲讲布局文件的优化。

加载布局文件,是必须在UI线程中完成的。我们通常是在onCreate方法中直调用setContentView,传入一个布局文件的id值,或者是通过LayoutInflater来将某一个布局文件转化成View对象。其实这两种方式的本质都是一样的,都是将xml文件转换成View对象。

  我们现在要做的事,就是如何让xml文件转换成View对象所花的时间最少。做到了这点,就可以很大程度的提高UI的流畅度。


优化布局, 减少布局的嵌套层级

a、使用drawableXXX属性

如果要实现这样一个效果,布局文件可以这样写

  1. <LinearLayout orientation="vertical">
  2. <ImageView/>
  3. <TextView/>
  4. </LinearLayout>

优化后:

  1. <TextView drawableBottom="@drawable/contact"/>

直接一个TextView就搞定,不需要在外面多一层LinearLayout


多使用RelativeLayout,少使用LinearLayout

如果这样的布局使用LinearLayout来做的话,那么会是以下这个效果

过多的LinearLayout嵌套LinearLayout,会造成UI加载的非常慢。这样的布局完全可以使用一个RelativeLayout来完成,里面的子元素根据相对于其他控件的位置即可确定。

嵌套使用LinearLayout很容易会导致视图层级过深。如果使用layout_weight这个参数不断的进行嵌套,有可能会让各个子View付出计算两次的代价

  1. <LinearLayout orientation="horizontal">
  2. <ImageView/>
  3. <LinearLayout orientation="vertical">
  4. <LinearLayout orientation="horizontal">
  5. <TextView/>
  6. <TextView/>
  7. </LinearLayout>
  8. <TextView/>
  9. <LinearLayout orientation="horizontal">
  10. <TextView/>
  11. <TextView/>
  12. </LinearLayout>
  13. </LinearLayout>
  14. </LinearLayout>

优化后代码:

  1. <RelativeLayout>
  2. <Image id=avatar layout_alignParentLeft=true />
  3. <TextView id=name layout_alignParentTop=true layout_toRightOf=@id/avatar />
  4. <TextView id=location layout_alignParentTop=true layout_toRightOf=@id/name />
  5. <TextView id=desc layout_below=@id/location layout_toRightOf=@id/avatar />
  6. ....
  7. </RelativeLayout>

使用merge标签

使用merge标签也是能够减少一些布局的层次。merge标签经常会和include标签相联系。

那么什么时候使用merge标签呢?下面举例子说明。

  1. <LinearLayout orientation="vertical">
  2. ......
  3. <include layout="@layout/include_view_layout"/>
  4. ......
  5. </LinearLayout>

而include_view_layout.xml 的代码如下:

  1. <LinearLayout orientation="vertical">
  2. <Button/>
  3. <Button/>
  4. </LinearLayout>

我们看到Button的父控件是LinearLayout,而include的父控件也是LinearLayout,这样子的布局最终的结果是

  1. LinearLayout orientation="vertical">
  2. <LinearLayout orientation="vertical">
  3. <Button/>
  4. <Button/>
  5. </LinearLayout>
  6. </LinearLayout>

里面的LinearLayout完全是多余,于是这时候,我们就可以在include_view_layout.xml文件中使用merge标签了。如下:

  1. <merge>
  2. <Button/>
  3. <Button/>
  4. </merge>

这样,在加载这个include标签的时候,系统会忽略merge标签,直接将merge标签内的元素添加到外层的LinearLayout去了,达到减少层级的效果。


延迟加载

在开发某些功能时候,有时候需要动态的根据条件来判断显示哪一个View,不显示哪一个View。一般的做法是将所有的View都写在布局文件中去,然后根据条件再来设置他们的可见度Visibility为GONE或者VISIBLE。这种做法逻辑简单,便于理解。

但是缺点就是那些不显示出来的View也占用了内存,消耗了inflate的时间。因为一个View,不论他的Visibility的值是什么,它都会被inflate出来,并占用内存空间。这时候其实就可以用到延迟加载的控件ViewStub了。

ViewStub是一个非常轻量级的控件,它占的资源非常小。注意,是ViewStub这个对象所占的资源小,但是你可以为ViewStub指定一个布局文件,这个布局文件被inflate的时候占的空间有可能很大。默认的情况下,ViewStub的所指定的布局文件是不被inflate的,只有当你调用了ViewStub的inflate方法时,ViewStub所指向的布局文件才会被inflate。所以ViewStub是一个延迟加载的控件。

  1. <LinearLayout
  2. android:orientation="vertical"
  3. android:layout_width="fill_parent"
  4. android:layout_height="fill_parent"
  5. android:gravity="center_horizontal">
  6.  
  7. <ViewStub
  8. android:id="@+id/viewstub1"
  9. android:layout_width="wrap_content"
  10. android:layout_height="wrap_content"
  11. android:layout="@layout/viewstub_layout1"/>
  12.  
  13. <ViewStub
  14. android:id="@+id/viewstub2"
  15. android:layout_width="wrap_content"
  16. android:layout_height="wrap_content"
  17. android:layout="@layout/viewstub_layout2"/>
  18. </LinearLayout>

在java代码中使用

  1. ViewStub stub1 = (ViewStub) findViewById(R.id.viewstub1);
  2. ViewStub stub2 = (ViewStub) findViewById(R.id.viewstub2);
  3. if(isLogin()) {
  4. stub1.inflate();
  5. } else {
  6. stub2.inflate();
  7. }

这样就不会有浪费资源空间去加载没必要的控件了。


减少inflate的次数

这个的典型例子就是ListView的优化。我们说ListView的优化,实际上说的就是Adapter中getView方法的优化,我们来看一个没有优化过的getView方法。

  1. @Override
  2. public View getView(int position, View convertView, ViewGroup parent) {
  3. MyItem product = list.get(position);
  4. convertView = getLayoutInflater()
  5. .inflate(R.layout.item_record, null);
  6.   TextView tvDate = (TextView) convertView
  7. .findViewById(R.id.tvDate);
  8. TextView tvYongtu = (TextView) convertView
  9. .findViewById(R.id.tvYongtu);
  10. TextView tvMoney = (TextView) convertView
  11. .findViewById(R.id.tvMoney);
  12. tvDate.setText(product.detaildate);
  13. tvYongtu.setText(product.auditmessage);
  14. tvMoney.setText(product.detailmoney);
  15. return convertView;
  16. }

  我们知道,ListView中的每一个Item被显示出来都要调用getView方法,这个Item如果滑出屏幕,又滑回来,重新显示在界面上的时候,又会再次调用getView方法。所以getView是不断的被调用的。而上面的代码,只要调用了getView方法,就一定会去inflate一个布局文件,真简直就是不敢想象的非常耗时的操作。于是,利用系统给我们的缓存convertView进行判断,可以大大减少inflate的次数。其实,findViewById也是一个很耗时的操作,我们可以利用ViewHolder来减少findViewById的次数。优化后的代码如下:

  1. @Override
  2. public View getView(int position, View convertView, ViewGroup parent) {
  3. MyItem product = list.get(position);
  4. ViewHolder holder;
  5. if (convertView == null) {
  6. convertView = getLayoutInflater().inflate(
  7. R.layout.item_record, null);
  8. holder = new ViewHolder();
  9. holder.tvDate = (TextView) convertView
  10. .findViewById(R.id.tvDate);
  11. holder.tvYongtu = (TextView) convertView
  12. .findViewById(R.id.tvYongtu);
  13. holder.tvMoney = (TextView) convertView
  14. .findViewById(R.id.tvMoney);
  15. convertView.setTag(holder);
  16. } else {
  17. holder = (ViewHolder) convertView.getTag();
  18. }
  19. holder.tvDate.setText(product.detaildate);
  20. holder.tvYongtu.setText(product.auditmessage);
  21. holder.tvMoney.setText(product.detailmoney);
  22. return convertView;
  23. }
  24. static class ViewHolder {
  25. TextView tvDate;
  26. TextView tvYongtu;
  27. TextView tvMoney;
  28. }

UI的流畅度优化的更多相关文章

  1. Android 滚动RecyclerView加载图片时的流畅度优化

    实现:使用onScrollStateChanged回调检测滚动状态,并在RecyclerViewAdapter内部设置类似isScrolling的状态值来控制网络图片的加载. 下面是代码举例: // ...

  2. iOS开发之多种Cell高度自适应实现方案的UI流畅度分析

    本篇博客的主题是关于UI操作流畅度优化的一篇博客,我们以TableView中填充多个根据内容自适应高度的Cell来作为本篇博客的使用场景.当然Cell高度的自适应网上的解决方案是铺天盖地呢,今天我们的 ...

  3. 转:iOS开发之多种Cell高度自适应实现方案的UI流畅度分析

    本篇博客的主题是关于UI操作流畅度优化的一篇博客,我们以TableView中填充多个根据内容自适应高度的Cell来作为本篇博客的使用场景.当然Cell高度的自适应网上的解决方案是铺天盖地呢,今天我们的 ...

  4. [原] Android持续优化 - 提高流畅度

    一.形象的感官一下流畅度概念 1. 这是官方给出的概念:Android流畅运行,需要运行60帧/秒, 则需要每帧的处理时间不超过16ms. 2. 每秒帧数,实际上就是指动画或视频每秒放映的画面数.因此 ...

  5. 利用Runloop优化流畅度

    我们可以对runloop添加观察者,当观察到状态为kCFRunLoopExit,kCFRunLoopBeforeWaiting的时候,做一些耗时的处理,废话不说,直接上代码 - (void)viewD ...

  6. iOS 优化界面流畅度的探讨

    界面流畅度 大都跟list scrollView有紧密关联 流畅的视觉:就是如丝般顺滑 不流畅视觉:”卡顿”,”抖动”,”迟顿感” 以上两种状态的描述 都是基于主观感觉,对于开发者来说 确实应该有一个 ...

  7. Android流畅度测试

    Android流畅度测试 测试方法一:系统自带-开发者模式 测试方法二:FPS Meter测试安卓帧数 H5页面加载速度:window.performance.timing 测试方法一:系统自带-开发 ...

  8. Android App性能评测分析-流畅度篇

    1.前言 在手机App竞争越来越激烈的今天,Android App的各项性能特别是流畅度不如IOS,安卓基于java虚拟机运行,触控响应的延迟和卡顿比IOS系统严重得多.一些下拉上滑.双指缩放快速打字 ...

  9. android流畅度比Ios差的原因

    Andorid更新了一个版本又一个版本,硬件从单核到双核到四核,系统流畅度总算基本能和iOS持平了.不过人们不禁会问,为什么都是基于Linux,两个系统会差别如此大?为什么iPhone 4用单核处理器 ...

随机推荐

  1. 4、nfs(存储服务器)

    1.NFS基本概述 NFS是Network File System的缩写及网络文件系统.NFS的主要功能是通过局域网络让不同的主机系统之间可以共享文件或目录. 通常中小企业首选NFS作为集群架构的存储 ...

  2. 如何使用ProcessOn制作思维导图

    新建一张思维导图之后你是不是有点茫然? 不是因为脑海里没思路,而是不知道怎么把脑海里的思路呈现出来?看到一个孤零零的中心主题和看起来有些简单的页面一时间有点无所适从? 很多人觉得思维导图好看但学起来难 ...

  3. 需求分析&系统设计

    这个作业属于哪个课程 课程链接 这个作业要求在哪里 作业要求 团队名称 朋友 代打了解一下 这个作业的目标 需求分析&系统设计 一.团队成员的姓名学号列表 学号 姓名 特长 061126 黄天 ...

  4. Easy Populate批量管理下载产品数据为空的解决办法

    把原来的先删除:http://aaaaacom/admin/easypopulate.php?langer=remove

  5. struts2使用注解的时候遇到的问题

    问题描述:     一切配置和注解都正确,但是无法访问action中的方法   解决问题:     之前将action层的包名命名成了web,改成action就对了   原因:     struts2 ...

  6. git fetch和pull的区别

    Git中从远程的分支获取最新的版本到本地有这样2个命令: 1. git fetch:相当于是从远程获取最新版本到本地,不会自动merge   1 2 3 Git fetch origin master ...

  7. C# class 浅拷贝 与 深拷贝

    MemberwiseClone 方法创建一个浅表副本,具体来说就是创建一个新对象,然后将当前对象的非静态字段复制到该新对象.如果字段是值类型的,则对该字段执行逐位复制.如果字段是引用类型,则复制引用但 ...

  8. easypoi 版本依赖关系

    <dependency> <groupId>org.apache.poi</groupId> <artifactId>poi</artifactI ...

  9. Java-DealString工具类

    import java.text.NumberFormat; import java.util.Date; import java.util.Locale; import java.util.Stri ...

  10. JavaScriptPlus操作类

    /// <summary> /// JavaScript 操作类 /// </summary> public class JavaScriptPlus { public Jav ...