Android布局性能调优

前言

在Android开发中,View的展示是最贴近用户,也是最能直观展示产品的手段。除了美观的界面之外,View的性能也是很重要的。

而View是由一层一层的View嵌套而成,形成类似于树的层级结构,通过层级结构展示View。View树的深度决定了展示的流畅度,深度越深,绘制需要的时间也就越长,体验效果越差。优化布局从另一方面说,也是就是想办法降低View树的深度,提高加载速度,达到性能调优。

下面介绍三个日常开发中可能会使用到的布局优化标签,以及一个实际开发中可能碰到的问题和解决方式。

代码比较简洁有的甚至没有,但是标签本来就是一些工具而已,把最后实战方面的看懂了,汲取了这种封装的思想,我写这篇文章的目的就达到了。

include标签

当一个View需要复用的时候,采取<include/>标签可以减少重复布局的使用。

使用场景

<include/>标签可能接触的都比较多,在Android布局中,很多时候会碰到需要使用相同的布局,例如每个Activity的TitleBar,评论框等。这个时候可以使用<include/>标签。

步骤

  1. 新建一个title.xml,在里面写好对应的布局文件的实现,
  1. 在需要使用的地方使用
other code...

<include layout="@layout/titlebar"/>

ViewStub标签

<ViewStub/>是一种不可见且大小为0的View,它的主要作用是当你不需要的时候不加载,当你需要的时候才去加载这个布局。也就是说<ViewStub/>实现了View的延迟加载。

需要注意的是:当ViewStub设置为可见或者被inflate之后,就会填充布局资源,之后会被填充的View给替代,和普通的View没有任何区别。

使用场景

错误图,也就是当数据错误的时候需要展示的一些图片和提示信息。但是这个错误图又不是每次都会显示,大部分情况下都是正常显示,只有当出现一些问题的时候才会显示错误图,这个时候就可以使用<ViewStub/>

步骤

  1. 在需要使用的地方使用<ViewStub/>标签
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent" >
 

    <ViewStub
        android:id="@+id/network_error_layout"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:layout="@layout/network_error"
        android:inflatedId="@+id/error_view"/>
 
</RelativeLayout>

2.新建network_error.xml文件

比如说一个ImageView

other code...

3.代码中使用

ViewStub stub = (ViewStub)findViewById(R.id.network_error_layout);
networkErrorView = stub.inflate();

在ViewStub中:

android:layout表示当需要展示的时候将会展示的Layout

android:inflatedId表示当在Java代码中调用ViewStub的inflate()或者setVisibility()方法返回的Id,也是就填充图Id。

merge标签

减少View树深度的利器

使用场景

在一般情况下,比如在ViewPager或者RecyclerView中,我们在每个Item的根布局中往往是使用LinearLayout或者其他的ViewGroup。如果有的时候我们只需要展示一张照片或者单独文字,这样就很不值得,因为每个ViewGroup对应的就在View树中又往深了一步。这个时候使用<merge/>标签来减少View树的深度再好不过。

merge标签可用于两种典型情况:

  • 布局顶结点是FrameLayout且不需要设置background或padding等属性,可以用merge代替,因为Activity内容视图的parent view就是个FrameLayout,所以可以用merge消除只剩一个。
  • 某布局作为子布局被其他布局include时,使用merge当作该布局的顶节点,这样在被引入时顶结点会自动被忽略,而将其子节点全部合并到主布局中。

步骤

<?xml version="1.0" encoding="utf-8"?>
<merge xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent" >
 
    <Button
        android:id="@+id/button"
        android:layout_width="match_parent"
        android:layout_height="@dimen/dp_40"
        android:layout_above="@+id/text"/>
 
    <TextView
        android:id="@+id/text"
        android:layout_width="match_parent"
        android:layout_height="@dimen/dp_40"
        android:layout_alignParentBottom="true"
        android:text="@string/app_name" />
 

优化实战例子

需求

在开发中有的时候TitleBar需要实现各种不同的布局,比如这个Activity需要一个TextView,这个Activity可能需要一个EditText,或者各种自定义View。

这个时候如果每个Activity的TItleBar都对应写新的布局,这样无疑加大了工作量。

思路(!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!)

我们可以将这个TitleBar很Activity逻辑分离,抽出布局作为titlebar.xml,里面包含了各种View。

然后再抽出一个TitleBar.class类,组合放在BaseActivity中。共通的东西可以在BaseActivity实现(比如左前方返回按钮),不同的东西放在子类中实现(EditText或者自定义View),灵活使用ViewStud实现延迟加载,merge减少层级。

代码结构可以使用Builder模式实现,注意<ViewStub/>和<merge/>的使用,注意空指针以及容错,这样就实现了一个标准TitleBar控件。

后话

还有一些很好的工具可以实现布局优化,比如Android Studio的lint,TraceView,HierarchyViewer,手机自带的开发人员选项中的工具等等等。

规范的布局代码加上工具检测,布局优化就OK了。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342

推荐阅读更多精彩内容