添加链接
link之家
链接快照平台
  • 输入网页链接,自动生成快照
  • 标签化管理网页链接
[Android技术专题]应用开发进阶必经之路之性能优化

[Android技术专题]应用开发进阶必经之路之性能优化

6 年前 · 来自专栏 张明云的知识共享

为了方便在手机上阅读,文章也会在公众号发出,更多原创文章和优质资源请关注公众号:


公众号:open_dev

更多Android技术资源交流请加群“ Android技术资源交流群 ”,第1期群分享精华已在微信公众号发出,本群的建群宗旨是分享优质的Android技术资源。群成员可以自由分享任何Android方面的技术资源和文章,并会不定期总结成文章方便大家阅读。群已满,请加群助理微信: Jf-1994 ,并注明原因是加Android技术资源交流群。

------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------


零、前言

性能优化在一款产品的迭代过程中非常重要;程序实现了功能、还原产品原型只能保证程序能用,但如果要让用户更愿意使用,产品得好用。试想一下如果你开发的产品启动慢、页面显示需要长时间转圈加载、页面切换卡顿、黑白屏、用一会机器就发烫、耗内存、OOM、程序切换到后台后占用内存无法释放......,这些问题就像正在玩游戏时弹出提示框这类糟糕的用户体验一样让用户恼火,如果用户不得不使用你的产品,可能还会一直忍受;但如果有很多同类竞品,糟糕的用户体验会大大影响留存率。有时候产品在市场上的表现差,真不能全怪产品和运营,程序体验问题也是很大一部分原因。

但大部分产品并没有足够重视性能问题,随便打开一个应用,即使是大厂出品,也极大可能存在过渡绘制和内存泄露的问题;也有可能是开发人员意识到了程序存在性能问题,但完成迭代就够忙的了,哪有时间去做这类不能体现绩效的事情。其实在越来越重视体验,同类产品竞争越来越激烈的环境下,对于开发人员来讲,只完成迭代,把功能做完远远不够,最重要的是把产品做好,让更多人愿意使用。重视性能问题,优化产品的体验,比改几个无关痛痒的bug会有价值得多。

网上能够找到很多关于性能优化很有价值的参考资料(详见文末),包括腾讯、阿里、魅族、豌豆荚、小米、UC等知名互联网公司都做过关于APP性能优化的分享,如果你专注于应用开发,并且想做一款备受欢迎的产品,性能优化是你进阶路上必须去学习和实践的。

一、性能问题分类

除了交互、视觉、内容方面的问题外,在用户使用过程中,给用户造成烦恼的问题都可以归结为是性能问题,比如上文中列出的这些都属于性能问题,按照影响的方面不同,可以分为如下几大类:

  • 内存问题: 耗内存、OOM、程序切换到后台后占用内存无法释放(OOM会影响产品的稳定性;耗内存、内存泄露会影响整机的性能;占用内存多预示着留给其它应用的剩余内存空间小);

  • 功耗问题: 发烫(耗电);

  • 流畅度问题: 启动慢、页面显示需要长时间转圈加载、页面切换卡顿、黑白屏(卡慢崩会让人烦躁);

针对上面一系列的性能问题,谷歌官方提供了各种各样的工具来针对性的解决各个方面的问题,也有很多不错的第三方工具值得尝试:

  • 内存问题: 提供了Android Studio的静态代码检测功能、Android Monitor;第三方内存泄露分析工具Leakcanary、MAT;

  • 功耗问题: 提供了GPU呈现模式、battery-historian、Android Monitor;

  • 流畅度问题: 提供了Android Studio的静态代码检测功能、Android Monitor、HierarchyViewer、StrictMode、过渡绘制检测工具、TraceView等;

除了上面提到的这些性能优化工具外,谷歌还在Youtube上提供了一系列关于Android应用性能优化的短视频 Android Performance Patterns ,介绍如何优化Android各个方面的性能问题。

二、性能优化指标

性能优化的效果仅凭感觉很难衡量,一切应该看数据说话,比如流畅度优化,刷新频率每秒越接近60帧越理想,但只要每秒钟超过24帧人眼就无法辨别了,所以仅凭感觉是无法区分优化前的30帧和优化后的40帧的区别的。为了说明做性能优化有足够的价值,就有必要通过一系列指标来说明优化前后的区别。

性能指标的定义应该具有可衡量、可比较的特点,所以每项性能指标可以是数值,也可以是一份报告,比如:

  • 流畅度: FPS,即Frams per Second,一秒内的刷新帧数,越接近60帧越好;

  • 启动时间: 时间,越短越好;

  • 内存泄露: AS静态代码检测结果、MAT检测结果,内存泄露很难用数值定义,但可以通过将优化前后工具检测的结果对比得出结论。没有内存泄露最好;

  • 内存大小: 峰值,峰值越低越好;

  • 功耗: 单位时间内的掉电量,掉电量越少越好;

从上面各项性能指标的定义可以看出,性能优化效果的评估主要是通过对比得出来的,性能如何只是相对的。只要针对同一个应用的同一项指标,优化后比优化前更优,就说明优化是有效果的。

三、性能优化原则和方法

1、优化原则

解决性能问题的过程中,遵循以下几个原则,有助于提高解决问题的效率:

  • 足够多的测量: 不要凭感觉去检测性能问题、评估性能优化的效果,应该保持足够多的测量,数据不会说谎。使用各种性能工具有助于快速定位问题,这比凭感觉要靠谱得多;

  • 使用低配置的设备: 同样的程序,在低端配置的设备中,相同的问题会暴露得更为明显;高配的设备很多时候会让你忽略掉性能问题;

  • 权衡利弊: 在能够保证产品稳定、按时交付的前提下去做优化,不能顾此失彼,为了性能优化导致产品迟迟不能交付;

2、优化方法

性能优化的指标很多,乍看上去无从下手,但和解bug一样,只要讲方法,事情就变得迎刃而解。对于大多数问题来讲,只要遵循了解问题→定位问题→分析问题→解决问题→验证问题的思路,基本上都可以解决:

  • 了解问题: 对于性能问题来讲,这个步骤只适用于某些明显的性能问题,很多无法感知的性能问题需要通过工具定位;

  • 定位问题: 通过工具检测、分析,定位在什么地方存在性能问题;如果很难定位,可以采用排除法(屏蔽部分代码,看现象是否仍然存在,如果还存在,则说明被屏蔽的代码没有问题,这样逐渐缩小问题的范围);

  • 分析问题: 找到问题后,分析针对这个问题该如何解决,确定解决方案;

  • 解决问题: 这个没什么可说的,如果是自己无法解决的问题,借助搜索引擎,你遇到过的问题很多人都遇到过,并且极有可能已经被解决了;

  • 验证问题: 保证每一次优化都有效,没有产生新问题,保证产品的稳定;

四、性能优化工具

本文重点介绍谷歌官方提供的一系列应用性能优化工具以及值得推荐的第三方性能优化工具,这些工具主要集中在如下几个地方:

  • 开发者选项: GPU呈现模式分析、GPU过渡绘制、严格模式、应用无响应ANR等;

  • IDE中: Android Studio,比如静态代码检测工具、Memory Monitor、CPU Monitor、NetWork Monitor、GPU Monitor、Layout Inspector、Analyze APK等;

  • SDK中: sdk\tools,比如DDMS、HierarchyViewer、TraceView等;

  • 第三方性能优化工具: MAT、Leakcanary等;

1、开发者选项:

首先从不需要依赖任何工具,直接借助手机中的开发者选项进行应用性能检测说起。开发者选项需要进入开发者模式后才能在系统设置中显示,对于大多数设备,可以通过如下方式在手机中开启开发者选项:打开“系统设置”→点击进入“关于手机”→连续点击“版本号”选项直至提示已进入“开发者模式”,就可以在“系统设置”中看到“开发者选项了”,打开“开发者选项”,可以看到很多能够帮助开发者检测应用性能的选项:


  • 调试GPU过渡绘制(Visualize GPU Overdraw): 过渡绘制用于检测你的程序是否存在不必要的绘制(举个栗子:同一个区域存在多个视图,刷新的时候被遮挡的视图也在绘制),导致显示时的性能问题,它可以帮助开发者解决如下问题:

(1)找出应用中哪些地方存在不必要的渲染;

(2)帮助开发者发现哪些地方可以减少渲染,提高程序运行效率;

显示过渡绘制区域的步骤如下:“开发者选项”→点击“调试GPU 过渡绘制”→点击“显示过渡绘制区域”,一旦使能,对设备中的任何应用都有效:


Android通过不同颜色来区分同一个区域绘制的次数,颜色越深,表示过渡绘制的次数越多,过渡绘制越严重。如下图所示,蓝色表示存在一次过渡绘制;深红色表示同一区域存在4次及以上的过渡绘制:

应用无法完全做到没有过渡绘制,优化是尽量避免不必要的过渡绘制,通常情况下保证同一区域过渡绘制少于三次都是合理的,即只要是出现红色(淡红色和深红色)的地方,就是需要优化的地方:

过渡绘制不仅仅会影响程序的刷新频率,还会导致程序启动慢、黑白屏、耗内存等问题,因为过渡绘制主要是因为布局复杂导致,android在加载布局文件的时候,实际上是读取xml文件并解析,然后根据每个视图的关系去测量、绘制、显示每一个视图;复杂的布局会需要更长的解析、测量、绘制、显示时间,也需要更多的内存(这与是否设置了视图背景有关)。在实际开发过程中,有如下几种常见的过渡绘制优化方法:

(1)使用merge标签: merge标签就是为减少布局层次而生的,它通过减少View树的层级来优化布局,merge只能作为xml布局的根标签使用(因为Activity的根布局是FrameLayout,所以只有Activity对应的布局文件根标签为FrameLayout时才适合使用merge标签),如果在代码中Inflate带merge标签的布局时,必须为这个自定义View指定一个父ViewGroup,并且设置attachToRoot为true。merge只能够在xml布局文件中使用,没有对应的java类。下面的实例演示了merge标签的用法,通过“GPU过渡绘制”查看优化前后的效果,可以明显看到通过merge标签解决了过渡绘制的问题;通过Hierarch View观察优化前后的视图树,可以明显看到使用merge标签后的视图层级减少了:

优化前:

布局文件:

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
             android:layout_width="fill_parent"
             android:layout_height="fill_parent"
             android:background="@android:color/white">
    <ImageView
        android:layout_width="fill_parent"
        android:layout_height="fill_parent"
        android:scaleType="center"
        android:src="@drawable/golden_gate"/>
    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center"
        android:padding="24dp"
        android:text="No Merge Layout"
        android:textColor="@android:color/black"/>
</FrameLayout>

效果图及过渡绘制显示:

Hierarchy View视图树:

优化后:

布局文件:

<?xml version="1.0" encoding="utf-8"?>
<merge xmlns:android="http://schemas.android.com/apk/res/android"
       android:background="@android:color/white">
    <ImageView
        android:layout_width="fill_parent"
        android:layout_height="fill_parent"
        android:scaleType="center"
        android:src="@drawable/golden_gate"/>
    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center"
        android:padding="24dp"
        android:text="Has Merge Layout"
        android:textColor="@android:color/black"/>
</merge>

效果图及过渡绘制显示:

Hierarchy View视图树:

(2)使用ViewStub标签: 在开发应用的时候,经常会遇到这样的情况,在程序运行时根据条件来决定显示/隐藏哪个视图;通常会在布局文件中将其写上去,默认隐藏,然后在代码中根据条件去判断是否显示。这样做的优点是逻辑清晰,但缺点是耗费资源,在布局文件中将某个视图默认设置为invisable或者gone,在Inflate布局文件的时候仍然会被infalte,同样会被实例化、设置属性,但有可能默认被隐藏的视图用户在某一次操作中很可能不会去触发它。为了提高布局文件加载效率和减少额外的资源消耗,强烈建议使用ViewStub标签,ViewStub是一个用于在运行时加载布局资源、不可见、宽高为0的View,在布局文件中使用它只是用于占位,在代码中没有手动加载它时,并不会影响页面的测量、绘制、显示效率,在代码中通过inflate加载ViewStub时,ViewStub会用在布局文件中为其指定的布局文件来代替它自身,通过前面的解释可想而知,ViewStub只能够被inflate一次,一旦加载后ViewStub对象就会被置为空;ViewStub标签有对应的java类ViewStub.java,通过阅读源码可以发现,确实在初始化的时候设置为隐藏、不绘制、宽高为0,并且它复写了View的dispatchDraw和draw方法,这俩方法是空方法,没有调用super的方法,也没有执行自己的代码:

public final class ViewStub extends View {
    public ViewStub(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
        super(context);
        final TypedArray a = context.obtainStyledAttributes(attrs,
                R.styleable.ViewStub, defStyleAttr, defStyleRes);
        mInflatedId = a.getResourceId(R.styleable.ViewStub_inflatedId, NO_ID);
        mLayoutResource = a.getResourceId(R.styleable.ViewStub_layout, 0);
        mID = a.getResourceId(R.styleable.ViewStub_id, NO_ID);
        a.recycle();
        setVisibility(GONE);
        setWillNotDraw(true);
    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        setMeasuredDimension(0, 0);
    @Override
    public void draw(Canvas canvas) {
    @Override
    protected void dispatchDraw(Canvas canvas) {

下面是ViewStub在Inflate前后的布局及视图树:

xml文件布局:

activity viewstub layout.xml:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    android:id="@+id/activity_view_stub_layout"
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">
    <include
        android:id="@+id/inclueId"
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="8"
        android:background="#ff00ff00"
        layout="@layout/inclue_viewstub_layout"/>
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="1"
        android:background="@android:color/black"/>
    <ViewStub
        android:id="@+id/viewStubId"
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="8"
        android:layout="@layout/inclue_viewstub_layout"/>
    <Button
        android:id="@+id/inflateViewStubBtnId"
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="3"
        android:text="InflateViewStub"
        android:onClick="onClick"
</LinearLayout>

inclue viewstub layout.xml:

<?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">
    <TextView
        android:id="@+id/titleTxtViewId"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_centerInParent="true"
        android:text="ViewStub!!!!"
        android:textColor="@android:color/black"
        android:textSize="24dp"/>
</RelativeLayout>

代码:

public void onClick(View v){
    switch(v.getId()){
        case R.id.inflateViewStubBtnId:{
            inflateViewStub();
        break;
private void inflateViewStub(){
    mViewStub = (ViewStub)findViewById(R.id.viewStubId);
    if(null == mViewStub){
        Toast.makeText(this, "ViewStub is Empty", Toast.LENGTH_LONG).show();
        return;
    mViewStub.inflate();

Inflate前:

界面效果:

Hierachy View:

Inflate后:

界面效果:

Hierachy View:

可以看到在ViewStub Inflate前ViewStub不占用布局层级,所以不会消耗程序资源;Inflate后会占用布局层级;在试验的过程中,点击两次及以上Inflate按钮时,会弹出“ViewStub is Empty”的Toast,说明mViewStub在实例化一次后再次实例化时会失败,因为在Inflate时已经被replace掉,系统找不到这个资源ID。

(3)使用Space: 过渡绘制问题是因为绘制引起的,space标签可以只在布局文件中占位,不绘制,Space标签有对应的java类Space.java,通过阅读源码可以发现,它继承至View.java,并且复写了draw方法,该方法为空,既没有调用父类的draw方法,也没有执行自己的代码,表示该类是没有绘制操作的,但onMeasure方法正常调用,说明是有宽高的。

xml布局:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:orientation="vertical">
    <TextView
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="2"
        android:gravity="center"
        android:text="TextView 1"
        android:textColor="@android:color/black"
        android:textSize="28dp"
        android:background="@android:color/darker_gray"/>
    <Space
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="1"
        android:background="@android:color/black"/>
    <TextView
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="2"
        android:gravity="center"
        android:text="TextView 2"
        android:textColor="@android:color/black"
        android:textSize="28dp"
        android:background="@android:color/darker_gray"/>
    <Space
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="1"/>
    <TextView
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="2"
        android:gravity="center"
        android:text="TextView 3"
        android:textColor="@android:color/black"
        android:textSize="28dp"
        android:background="@android:color/darker_gray"/>
</LinearLayout>

效果图:

可以看到在布局中给第一个Space控件设置了黑色背景,但从效果图可以看出Space并没有变成黑色,说明没有执行绘制方法。

Hierarchy View:

观察视图树也可以看出Space会占用空间(因为有宽高)。

(4)去掉不必要的背景: 如果不是通过测量和仔细分析,你很难发现这个不经意的细节会是导致过渡绘制、内存问题的主要原因,每个Activity都会在AndroidManifest.xml中设置主题,主题的目的是设置界面的显示风格,但在设置主题的时候通常情况下默认给Window设置了背景,注意是Window而不是Activity,Activity是依附在Window上的,Android系统在刷新整个界面时不仅仅是刷新Activity,还会刷新Window。如果默认没有去掉window的背景,并且在布局文件中给Activity设置了背景,就会存在过渡绘制的问题,具体情况可以看下面的实例:

activity background layout.xml(这里为了演示在布局文件中为每个视图设置了背景,在真实情况中没有必要为每个视图都设置):

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
    android:id="@+id/activity_background_layout"
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@android:color/white">
    <ScrollView
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:layout_marginLeft="20dp"
        android:layout_marginRight="20dp">
        <TextView
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:padding="10dp"
            android:background="@android:color/white"
            android:text="@string/more_text"/>
    </ScrollView>
</RelativeLayout>

使用该布局的Activity对应的主题如下:

<style name="AppTheme" parent="@android:style/Theme.Light.NoTitleBar">
</style>

通过过渡绘制工具检测,存在过渡绘制:

下面是去掉window背景后的效果(布局文件不变,主题变动如下):

<style name="AppTheme" parent="@android:style/Theme.Light.NoTitleBar">
    <item name="android:windowBackground">@null</item>
</style>

效果图:

说明:

1、在主题中去掉Window的背景时要注意,去掉之后必须重新运行程序检查一下,避免有些Activity并没有设置背景导致界面背景为黑色;

2、有的程序为了避免冷启动时界面黑屏/白屏的问题,在主题中为window设置了一张图片,然后在布局文件中为Activity也设置了背景,这样既会导致过渡绘制问题,还会导致内存问题(同一个页面两张全屏的图片,双倍内存);所以这种解决方式并不妥,如果是启动速度问题,直接优化启动速度比这种方式靠谱。

(5)其它:

1、通过Canvas的clipRect方法控制每个视图每次刷新的区域,这样可以避免刷新不必要的区域,从而规避过渡绘制的问题;

2、如对一个View做Alpha转化,需要先将View绘制出来,然后做Alpha转化,最后将转换后的效果绘制在界面上。通俗点说,做Alpha转化就需要对当前View绘制两遍,可想而知,绘制效率会大打折扣,耗时会翻倍;

3、多和产品、UI沟通,避免过渡设计,过渡绘制最好的解决方案是界面的布局本身就不复杂,这样程序实现出来的界面就会少很多过渡绘制,优化也会简单很多。

总结一下过渡绘制的检测和解决方案:通过“开发者选项”中的“显示过渡绘制”和Android提供的工具“HierarchyViewer”,以每个界面为单位,可以完全检测出每个界面的过渡绘制问题;因为导致过渡绘制的原因不一,所以也会有多种对应的解决方案:

1、merge标签可以解决相同布局嵌套导致的过渡绘制问题;

2、ViewStub标签可以解决动态加载页面布局,避免默认加载不必要布局的问题;

3、Space标签可以解决只占位、不刷新的视图问题;

4、去掉Window背景可以解决所有界面的过渡绘制问题;

5、clipRect可以解决只刷新固定区域的问题;

6、不必要的alpha值设置可以解决同一视图被多次绘制的问题;

7、最重要的是产品设计合理,多和产品、UI沟通,避免无意义的工作。

  • GPU呈现模式分析(Profiling GPU Rendering):

从Android 4.1开始,在“开发者选项”中提供了GPU呈现模式分析的选项,GPU呈现模式是一个方便你快速观察UI渲染效率的工具,主要作用是实时查看每一帧的渲染效率,定位哪里存在渲染的性能问题;通过如下方式可以打开GPU呈现模式分析:“系统设置”→“开发者选项”→“GPU呈现模式分析”→在弹出的窗口中选择“在屏幕上显示成条形图(On screen as bars)”。

打开GPU呈现模式后,你可以在机器的任何界面看到如下图所示的条形图,顶部通知栏、当前活动程序(主窗口)、底部导航栏都会有对应的呈现模式条形图,用于观察通知栏、当前活动界面、导航栏的渲染效率。

随着界面的刷新,界面上会滚动显示锤子的柱状图来表示每帧画面说需要的渲染时间,柱状图越高表示花费的渲染时间越长。中间有一根绿色的横线,代表每帧的最长渲染时间:16ms,我们需要确保每一帧花费的总时间都低于这条横线,这样才能够避免出现卡顿的问题。

从图中可以看出,每一条柱状线包含三种颜色,但从Android 6.0开始,你看到的每条柱状线已不止三种颜色:

每种颜色代表每一帧渲染过程中需要完成的某一件事情,因为6.0之前的三种颜色不大能够清晰地帮助我们定位性能问题的具体原因,所以从6.0开始,将每一帧的渲染过程拆分成了8个步骤,每个步骤一种颜色,每种颜色的意义如下:

(1)Swap Buffers: 表示处理任务的时间,也可以说是CPU等待GPU完成任务的时间,线条越高,表示GPU做的事情越多;

(2)Command Issue: 表示执行任务的时间,这部分主要是Android进行2D渲染显示列表的时间,为了将内容绘制到屏幕上,Android需要使用Open GL ES的API接口来绘制显示列表,红色线条越高表示需要绘制的视图更多;

(3)Sync & Upload: 表示的是准备当前界面上有待绘制的图片所耗费的时间,为了减少该段区域的执行时间,我们可以减少屏幕上的图片数量或者是缩小图片的大小;

(4)Draw: 表示测量和绘制视图列表所需要的时间,蓝色线条越高表示每一帧需要更新很多视图,或者View的onDraw方法中做了耗时操作;

(5)Measure/Layout: 表示布局的onMeasure与onLayout所花费的时间,一旦时间过长,就需要仔细检查自己的布局是不是存在严重的性能问题;

(6)Animation: 表示计算执行动画所需要花费的时间,包含的动画有ObjectAnimator,ViewPropertyAnimator,Transition等等。一旦这里的执行时间过长,就需要检查是不是使用了非官方的动画工具或者是检查动画执行的过程中是不是触发了读写操作等等;

(7)Input Handling: 表示系统处理输入事件所耗费的时间,粗略等于对事件处理方法所执行的时间。一旦执行时间过长,意味着在处理用户的输入事件的地方执行了复杂的操作;

(8)Misc Time/Vsync Delay: 表示在主线程执行了太多的任务,导致UI渲染跟不上vSync的信号而出现掉帧的情况;出现该线条的时候,可以在Log中看到这样的日志:

I/Choreographer(*): Skipped XXX frames! The application may be doing too much work on its main thread