使用TraceView找到卡顿的元凶以及StrictMode优化代码逻辑

读完本文你将了解到:
1.TraceView 是什么;
2.如何生存trace文件;
3.根据TraceView显示定位问题
4.解决DDMS中find功能无法使用的问题
5.TraceView使用场景
6.StrictMode严苛模式使用

1.TraceView 是什么

TraceView 是 Android SDK 中内置的一个工具,它可以加载 trace 文件,用图形的形式展示代码的执行时间、调用次数及调用栈,便于我们分析。

trace 文件是 log 信息文件的一种,可以通过代码,Android Studio,或者 DDMS 生成。

使用 Android SDK 提供的工具可以生成很多 log 文件,便于我们分析当前应用的内存、布局等状况,下面是几种文件的截图:

log文件

手机卡顿很多时候都是由于某个操作过于耗时,在茫茫代码中查找元凶未免太过痛苦,这时候就该体现 TraceView 的价值了。

2.生成 trace 文件

生成 trace 文件有三种方法:

  • 使用代码生成
 Debug.startMethodTracing("shixintrace"); //开始 trace保存文件到 "/sdcard/shixintrace.trace"
    // ...
    Debug.stopMethodTracing();    //结束

代码很简单,当你调用开始代码的时候,系统会生产 trace 文件,并且产生追踪数据,当你调用结束代码时,会将追踪数据写入到 trace 文件中。

下一步使用 adb 命令将 trace 文件导出到电脑:

adb pull /sdcard/shixintrace.trace /tmp

使用代码生成 trace 方式的好处是容易控制追踪的开始和结束,缺点就是步骤稍微多了一点。

  • 使用 Android Studio生成

Android Studio 内置的 Android Monitor 可以很方便的生成 trace 文件到电脑。

在 CPU 监控的那栏会有一个闹钟似的的按钮,未启动应用时是灰色:

Android Monitor

启动应用后,这个按钮会变亮,点击后开始追踪,相当于代码调用 startMethodTracing:

startMethodTracing

当要结束追踪时再次点击这个按钮,就会生成 trace 文件了。

生成 trace 后 Android Studio 自动加载的 traceview 图形如下:

traceview分析图

从这个图可以大概了解一些方法的执行时间、次数以及调用关系,也可以搜索过滤特定的内容。

左上角可以切换不同的线程,这其实也是直接用 Android Studio 查看 trace 文件的缺点:无法直观地对比不同线程的执行时间。

鼠标悬浮到黄色的矩形上,会显示对应方法的开始、结束时间,以及自己占用和调用其他方法占用的时间比例:

具体信息
  • 使用 DDMS生成

DDMS 即 Dalvik Debug Monitor Server ,是 Android 调试监控工具,它为我们提供了截图,查看 log,查看视图层级,查看内存使用等功能,可以说是如今 Android Studio 中内置的 Android Monitor 的前身。

双击 shift 弹出全局搜索,搜索 “Android Device Monitor”:

示例图

或者直接在 设置里设置 Android Device Monitor 的快捷键:

示例图

打开 Android Device Monitor,在 DDMS 中打开 trace 文件,DDMS 会启动 TraceView 加载 trace 文件:

trace分析

上图介绍了 TraceView 的大致内容:

  • 上半部分显示了不同线程的执行时间
    • 其中不同的颜色表示不同的方法
    • 同一个颜色越长,说明执行时间越久,如图中的主线程 main
    • 空白表示这个时间段内没有执行内容
  • 下半部分展示了不同方法的执行时间信息,关键指标有三个:
    • Cpu Time/Call :该方法平均占用 CPU 的时间
    • Real Time/Call :平均执行时间,包括切换、阻塞的时间,>= Cpu Time
    • Calls + Recur Calls/Total :调用、递归次数

点击下面的任意一个方法,可以看到它的详细信息:

  • Parents:选中方法的调用处
  • Children:选中方法调用的方法
详细信息

3.根据 TraceView信息显示内容定位问题

定位问题时, TraceView 的使用方式:

  • 从上半部分查看哪些线程执行时间长?什么时候开始执行?与主线程交错时间?
  • 哪些方法的执行需要花费很长时间
    • 点击 TraceView 中的 Cpu Time/Call,按照占用 CPU 时间从高到低排序
  • 哪些方法调用次数非常频繁
    • 点击 TraceView 中的 Calls + Recur Calls/Total ,按照调用次数从高到底排序

排序后,然后逐个排查是否有项目代码或者依赖库代码,有的话点击查看详情,查看是这个方法还是调用的子方法的问题,进一步定位问题。

4.解决 DDMS 中的 TraceView 无法搜索,find 无法使用的问题

Traceview 中信息太多,想要查找可以使用最下方的 find:

find

但是目前 DDMS 中的 TraceView 有 bug,find 无法使用, 解决办法就是直接使用 SDK 中的 TraceView。

直接打开 SDK 中的 TraceView :

TraceView

然后打开之前生成的 trace 文件:

trace文件分析

如果直接打开 traceview 有问题,可以通过命令行 traceview 打开:

命令行方式打开tarceview

虽然提示 deprecated,但起码在搜索上还是比 Android Device Monitor 中好用。

5.TraceView 的使用场景

在发现某个页面或者操作会卡顿时,可以使用 TraceView 定位问题代码。

比如启动,加载图片列表卡顿等情况。

还有一点,不要过早优化!

6.代码逻辑优化-StrictMode简单介绍

  • 1.简介
    StrictMode类是Android 2.3 (API 9)引入的一个工具类,可以用来帮助开发者发现代码中的一些不规范的问题,以达到提升应用响应能力的目的。举个例子来说,如果开发者在UI线程中进行了网络操作或者文件系统的操作,而这些缓慢的操作会严重影响应用的响应能力,甚至出现ANR对话框。为了在开发中发现这些容易忽略的问题,我们使用StrictMode,系统检测出主线程违例的情况并做出相应的反应,最终帮助开发者优化和改善代码逻辑

  • 2.用途
    使用严格模式,系统检测出主线程违例的情况会做出相应的反应,如日志打印,弹出对话框亦或者崩溃等。换言之,严格模式会用不同的策略对应用的违例细节做监控,通过一定的方式暴露给开发者方便优化与改善。

  • 3.使用
    通常在Activity或者自定义的Application类的onCreate方法中启动StrictMode.

// DEVELOPER_MODE代表只在调试模式下启用StrictMode

if (DEVELOPER_MODE) {

   //线程方面的策略
   StrictMode.setThreadPolicy(new StrictMode.ThreadPolicy.Builder()

           .detectDiskReads()

           .detectDiskWrites()

           .detectNetwork()   // or .detectAll() for all detectable problems

           .penaltyLog()

           .build());

   // VM方面的策略
   StrictMode.setVmPolicy(new StrictMode.VmPolicy.Builder()

           .detectLeakedSqlLiteObjects()

           .detectLeakedClosableObjects()

           .penaltyLog()

           .penaltyDeath()

           .build());
}

StrictMode通过策略方式来让你自定义需要检查哪些方面的问题,主要有两种策略:

  • ThreadPolicy:线程方面的策略
    StrictMode.ThreadPolicy.Builder主要方法解释:
detectNetwork():用于检查UI线程中是否有网络请求

detectDiskReads()和detectDiskWrites():磁盘读写检查

detectCustomSlowCalls():主要用于帮助开发者发现UI线程中调用的哪些方法执行的比较慢,
要和StrictMode.noteSlowCall配合使用,只有通过StrictMode.noteSlowCall标记“可能会”执行
比较慢的方法,只有标记过的方法才能被检测到,日志中会记录方法的执行时间
(注意:只有在主线程中执行的方法才会显示执行时间,在其他线程中执行的方法,
就算是使用StrictMode.noteSlowCall标记,在日志中也不会打印执行时间)

  • VmPolicy:VM方面的策略
    StrictMode.VmPolicy.Builder()主要方法解释:
detectActivityLeaks(): 用户检查 Activity 的内存泄露情况

detectLeakedClosableObjects() 和 detectLeakedSqlLiteObjects():资源没有正确关闭时回触发
detectLeakedSqlLiteObjects() 和 detectLeakedClosableObjects()的用法类似,
         只不过是用来检查 SQLiteCursor 或者 其他 SQLite 对象是否被正确关闭

detectLeakedRegistrationObjects() 用来检查 BroadcastReceiver或者ServiceConnection注册
类对象是否被正确释放

setClassInstanceLimit(),设置某个类的同时处于内存中的实例上限,可以协助检查内存泄露

惩罚:

penaltyDeath():当触发违规条件时,直接Crash掉当前应用程序

penaltyDeathOnNetwork():当触发网络违规时,Crash掉当前应用程序

penaltyDialog():触发违规时,显示对违规信息对话框

penaltyFlashScreen():会造成屏幕闪烁,不过一般的设备可能没有这个功能

penaltyDropBox():将违规信息记录到 dropbox 系统日志目录中(/data/system/dropbox),
    你可以通过如下命令进行插件

adb shell dumpsys dropbox dataappstrictmode --print

permitCustomSlowCalls()、permitDiskReads ()、permitDiskWrites()、
permitNetwork: 如果你想关闭某一项检测,可以使用对应的permit*方法

最后:我们也可以在Android设备的设置(Settings)中启用StrictMode(亲测没有什么效果):开发者选项 > 启用严格模式。

最后分享一个小例子:

public class MainActivity extends AppCompatActivity{
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        StrictMode.setThreadPolicy(new StrictMode.ThreadPolicy.Builder()
                .detectAll()
                .penaltyLog()
                .build());

        findViewById(R.id.btn).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                writeToExternalStorage();
            }
        });
    }

    /**
     * 文件系统的操作
     */
    public void writeToExternalStorage() {
            try {
                File externalStorage = Environment.getExternalStorageDirectory();

                File mbFile = new File(externalStorage, "xxx.txt");
                if (mbFile.exists()){
                    mbFile.createNewFile();
                }
            OutputStream output = new FileOutputStream(mbFile, true);
            output.write("www.wooyun.org".getBytes());
            output.flush();
            output.close();
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

使用penaltyLog():

D/StrictMode: StrictMode policy violation; ~duration=3 ms: 
android.os.StrictMode$StrictModeDiskWriteViolation: policy=65567 violation=1
at android.os.StrictMode$AndroidBlockGuardPolicy.onWriteToDisk(StrictMode.java:1253)
at java.io.FileOutputStream.<init>(FileOutputStream.java:220)
at com.perf.strictmodedemo.MainActivity.writeToExternalStorage(MainActivity.java:44)
at com.perf.strictmodedemo.MainActivity$1.onClick(MainActivity.java:28)
at android.view.View.performClick(View.java:5619)
at android.view.View$PerformClick.run(View.java:22298)
at android.os.Handler.handleCallback(Handler.java:754)
at android.os.Handler.dispatchMessage(Handler.java:95)
at android.os.Looper.loop(Looper.java:165)
at android.app.ActivityThread.main(ActivityThread.java:6365)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:883)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:773)

使用penaltyDialog():

penaltyDialog

总结:StrictMode适合作为一些基础规范问题排查。

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,918评论 25 707
  • 用两张图告诉你,为什么你的 App 会卡顿? - Android - 掘金 Cover 有什么料? 从这篇文章中你...
    hw1212阅读 12,709评论 2 59
  • 在使用app过程,尤其是在浏览器内,难免会出现浏览网页的时候出现卡顿的过程,那这类问题怎么分析、定位、解决问题?这...
    jb2阅读 9,725评论 0 6
  • 求救 文/墨染 钻心地哀鸣 撕心裂肺地,响彻在 浓浓的雾里 寒风,敲打我身体 像绣花针一样,穿透 穿透我的皮肤 和...
    __墨染阅读 201评论 4 9
  • 李花香,寄语春风,诉尽满城烟雨。 烟花艳,空燃灿烂,划落彼岸清秋曲。 听一曲,瑟瑟琴箫,浅了昨日愿。 情何许,问向...
    野生运营笔记阅读 257评论 0 0