强大的可视化利器 Chrome Trace Viewer 使用详解

2024-06-05 09:38

本文主要是介绍强大的可视化利器 Chrome Trace Viewer 使用详解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

image|690x398

最近研究了下 Chrome 自带的 Trace Viewer,发现功能还挺强大的,用来做 Performance Profile,Timeline Tracing 等很方便,但官方的使用文档比较晦涩,资料也比较匮乏和分散,就专门整理了下。

注:Trace Viewer 其实可以脱离 chrome://tracing 在线使用,具体操作后面会讲到

Trace Viewer 介绍

Trace-Viewer is the javascript frontend for Chrome about:tracing and Android systrace. providing rich analysis and visualization capabilities for trace files, supporting both the linux kernel trace format and Chrome’s base/trace_event.

它是一个强大的可视化展示和分析工具,之前 google 有一个专门的 trace-viewer 项目,现在该项目合并到了 catapult 中, catapult 是 Chromium 工程师开发的一系列性能工具的合集,可以用来收集、展示、分析 Chrome、Website 甚至 Android 的性能。

catapult 源码里包含了很多 test case,git clone 下来后,运行 $CATAPULT/bin/run_dev_server. 然后在浏览器中打开 http://localhost:8003/tracing_examples/trace_viewer.html 能看到很多 Demo

Untitled 4 (3)|635x500

不过 Google 目前正在开发另一套工具 perfetto 打算替换 Trace Viewer,我初步使用了下,感觉不如 Trace Viewer 简洁。

在开源项目中的使用

Golang 用它来可视化展示执行过程 https://golang.org/cmd/trace/

go test -trace trace.out pkg
go tool trace trace.out

tracer-lock|690x284

Facebook 的 Buck Build 系统,用它来展示 Build 时发生了什么

buck_chrome_sample|619x500

从这两个项目可以看出来 Trace Viewer 对于呈现「某段时间发生了什么」 还是有过人之处。

Trace Viewer Format

Trace Viewer 有一套自己的 Trace Event Format,只要文件遵循这个格式,就可以被展示。比如:

image|690x62

对应的 json 文件内容:

[ {"name": "出方案", "ph": "B", "pid": "Main", "tid": "工作", "ts": 0},{"name": "出方案", "ph": "E", "pid": "Main", "tid": "工作", "ts": 28800000000}, {"name": "看电影", "ph": "B", "pid": "Main", "tid": "休闲", "ts": 28800000000},{"name": "看电影", "ph": "E", "pid": "Main", "tid": "休闲", "ts": 32400000000},{"name": "写代码", "ph": "B", "pid": "Main", "tid": "工作", "ts": 32400000000},{"name": "写代码", "ph": "E", "pid": "Main", "tid": "工作", "ts": 36000000000},{"name": "遛狗", "ph": "B", "pid": "Main", "tid": "休闲", "ts": 36000000000},{"name": "遛狗", "ph": "E", "pid": "Main", "tid": "休闲", "ts": 37800000000}
]

每一个 Event 主要由以下几部分组成:

{"name": "myName", // 事件名,会展示在 timeline 上"cat": "category,list", // 事件分类,类似 Tag,但 UI 上不支持选择 Tag"ph": "B", // phase,后面着重会讲到"ts": 12345, // 事件发生时的时间戳,以微秒表示"pid": 123, // 进程名"tid": 456, // 线程名"args": { // 额外参数,当选中某个 event 后,会在底部的面板展示"someArg": 1,"anotherArg": {"value": "my value"}}
}

其中最重要的是 ph,最常用的组合是 B 和 E,分别表示 Begin 和 End,有了这两个信息,Trace Viewer 就能在 timeline 上找到起止点,将它绘制出来。也可以简化为 X,然后加上 dur 表示 duration

[ {"name": "出方案", "ph": "X", "pid": "Main", "tid": "工作", "ts": 0, "dur": 28800000000},{"name": "看电影", "ph": "X", "pid": "Main", "tid": "休闲", "ts": 28800000000, "dur": 360000000},{"name": "写代码", "ph": "X", "pid": "Main", "tid": "工作", "ts": 32400000000, "dur": 360000000},{"name": "遛狗", "ph": "X", "pid": "Main", "tid": "休闲", "ts": 36000000000, "dur": 180000000},
]

但颜色看起来有点随机,能不能自定义呢,可以的,这时就要用到 cname 这个参数:

 {"name": "出方案", "cname": "good", "ph": "X", "pid": "Main", "tid": "工作", "ts": 0, "dur": 28800000000},

将 cname 设置为 good,视觉上会看到绿色,在这里有可选的 cname 列表(藏的真够深···)

左边的 Process Main 能不能也进行个性化定制以显得不那么突兀呢?比如将它改成「时间线」

image|690x70

这就要使用 MetaData Phase 了

{"name": "process_name", "ph": "M", "pid": "Main", "tid": "工作", "args": {"name": "时间线"}}

它的表示方式是将 ph 设为 M,然后将 name 设为 5 种可选值中的一种,比如 process_name 则表示要对进程名做一些定制。metadata 主要是用来为某一类 event 提供更多的信息,比如自定义名字或重新设置排序优先级。

有一些事件,它不是执行耗时,但包含了其他事件,比如「我的一天」,它不是一个可被执行的事件,但将一天做的事情包含进来。这些被称为「异步事件」,表现上就像这样:

image|690x85

独占一栏,同时使用了斜体字,如果仔细看,还会发现顶部多了一条黑色的线。异步事件的 ph 为 b 和 e 也是 Begin 和 End,只不过变成了小写,同时还要提供 id 和 cat

[{"name": "我的一天", "ph": "b", "cat": "daily", "id": "my-day", "pid": "Main", "tid": "工作", "ts": 0},{"name": "我的一天", "ph": "e", "cat": "daily", "id": "my-day", "pid": "Main", "tid": "工作", "ts": 37800000000}, 
]

id 用来唯一标识两个异步事件是不是应该被作为同一个来看待(有可能多个异步事件名字都一样),cat 也是必选项,这个不太理解,可能等待的时间太孤独,需要一只猫来陪伴吧。

一件事可能会被拆分为更小的单元,然后执行过程中也会穿插着其他的事件,针对这样的场景,可以通过 Flow Event 来表示

Untitled 4|690x91

注意,在「出方案」和「写代码」之间多了一条线

  {"name": "出方案", "cname": "good","ph": "X", "pid": "Main", "tid": "工作", "ts": 0, "dur": 28800000000}, {"name": "connect", "ph": "s", "id": "my-work", "pid": "Main", "tid": "工作", "ts": 28800000000}, {"name": "connect", "ph": "f", "bp": "e", "id": "my-work", "pid": "Main", "tid": "工作", "ts": 32400000000},{"name": "写代码", "ph": "X", "pid": "Main", "tid": "工作", "ts": 32400000000, "dur": 3600000000},

Flow Event ph 的开始和结束分别为 s(start) 和 f(finish),跟 Async Event 有点像,但有几点不同:

  • 不需要带上 cat
  • 同一个 Flow Event,除了 id 相同外,名字也必须相同
  • 结束时用 "bp": "e" 来表示

Flow Event 需要依附在已有的 Event 上,依附的过程其实就是 ts 匹配,相同/最近 ts 的 Event 会自动被匹配到。

出方案的过程中,可能会有纪念意义的事件需要记录下,比如想到了某个点子,这时可以用 ph: "I" 来表示(Instant)

{"name": "想到一个点子", "ph": "I", "pid": "Main", "tid": "工作", "ts": 18800000000}

Untitled 4 (1)|690x106

中间那根红线就是我们刚加的事件,选中之后会在底部出现该事件的详细信息。

作为一个上进的青年,一天下来之后,需要留一段时间反思今天做的好的和不好的,这可以通过 Snapshot 来体现,就像这样

image|690x146

右上角那个圆点就是 Snapshot 在 Timeline 上的表现,点击之后可以在底部看到详细内容。

{"name": "反思", "ph": "O", "id": "Ref", "pid": "Main", "tid": "其他", "ts": 37800000000, "args": {"snapshot": "充实的一天呢"}}

Snapshot 的 ph 为 O (字母 O),没有什么限制,详细的内容往 args.snapshot 里放就可以了。

这些差不多就是常用的 ph 了,部分 ph 会有附带的规则,再简单总结下:

ph B/E // 正常的开始/结束事件,最常见,也可以用 X + dur 来表示
ph M // Metadata 用来对一类 Event 附加更详细的信息,可以带来 UI 上的变化
ph I // 瞬时事件,类似 Mark 一下
ph s/e // 异步事件,表示自定义的一个事件,表现上跟正常事件会有区别
ph s/f // Flow 事件,会出现箭头,要通过 ts 匹配最近的 event,结束要使用 bp: e
ph O // Snapshot,表现上是一个醒目的圆点,可以在 `args.snapshot` 里放任意数据

以下是 Demo 对应的 Event 内容,只要能生成这样的 JSON 文件,就可以使用 Trace Viewer 强大的可视化能力。

[ {"name": "我的一天", "ph": "b", "cat": "daily", "id": "my-day", "pid": "Main", "tid": "工作", "ts": 0},{"name": "我的一天1", "ph": "e", "cat": "daily", "id": "my-day", "pid": "Main", "tid": "工作", "ts": 37800000000}, {"name": "出方案", "cname": "good","ph": "X", "pid": "Main", "tid": "工作", "ts": 0, "dur": 28800000000}, {"name": "想到一个点子", "ph": "I", "pid": "Main", "tid": "工作", "ts": 18800000000}, {"name": "connect", "ph": "s", "id": "my-work", "pid": "Main", "tid": "工作", "ts": 28800000000}, {"name": "connect", "ph": "f", "bp": "e", "id": "my-work", "pid": "Main", "tid": "工作", "ts": 32400000000},{"name": "写代码", "ph": "X", "pid": "Main", "tid": "工作", "ts": 32400000000, "dur": 3600000000},{"name": "反思", "ph": "O", "id": "Ref", "pid": "Main", "tid": "其他", "ts": 37800000000, "args": {"snapshot": "充实的一天呢"}},{"name": "看电影", "ph": "B", "pid": "Main", "tid": "休闲", "ts": 28800000000},{"name": "看电影", "ph": "E", "pid": "Main", "tid": "休闲", "ts": 32400000000},{"name": "遛狗", "ph": "X", "pid": "Main", "tid": "休闲", "ts": 36000000000, "dur": 1880000000},{"name": "process_name", "ph": "M", "pid": "Main", "tid": "工作", "args": {"name": "时间线"}}
]

在线使用 Trace Viewer

通过 chrome://tracing 的方式来使用 Tracer Viewer 还是不太方便,也不利于传播,Google 虽然在 catapult 里提供了 trace2html,但包含的文件很多,使用起来还是有点麻烦,于是参考了 go trace 的源码,把相关文件上传到了 CDN,然后在一个 html 文件里引用,这样只需一个文件即可。

<!DOCTYPE html>
<!--
Copyright (c) 2014 The Chromium Authors. All rights reserved.
Use of this source code is governed by a BSD-style license that can be
found in the LICENSE file.
--><head><meta http-equiv="Content-Type" content="text/html; charset=UTF-8" /><style>body {font-family: arial, sans-serif;}</style><scriptsrc="https://sf1-ttcdn-tos.pstatp.com/obj/developer-baas/baas/ttnbrzs5vgcryya2z2/7d1b75402231d1cd_1583755520315.js"></script><script>'use strict';window.__hideTraceViewerPolyfillWarning = true;function onTraceViewerImportFail() {document.addEventListener('DOMContentLoaded', function () {document.body.textContent ='tracing/bin/trace_viewer_full.html is missing. ' +'Run vulcanize_trace_viewer from $TRACE_VIEWER and reload.';});}</script><link rel="import"href="https://sf1-ttcdn-tos.pstatp.com/obj/developer-baas/baas/ttnbrzs5vgcryya2z2/4b94a178762f4049_1583756222365.html"onerror="onTraceViewerImportFail(event)"><style>html,body {box-sizing: border-box;overflow: hidden;margin: 0px;padding: 0;width: 100%;height: 100%;}#trace-viewer {width: 100%;height: 100%;}#trace-viewer:focus {outline: none;}</style><script>'use strict';(function () {var viewer;var url;var model;function load() {var req = new XMLHttpRequest();var isBinary = /[.]gz$/.test(url) || /[.]zip$/.test(url);req.overrideMimeType('text/plain; charset=x-user-defined');req.overrideMimeType('contentType: "application/x-www-form-urlencoded;charset=utf-8"');console.log(url);req.open('GET', url, true);if (isBinary)req.responseType = 'arraybuffer';req.onreadystatechange = function (event) {if (req.readyState !== 4)return;window.setTimeout(function () {if (req.status === 200)onResult(isBinary ? req.response : req.responseText);elseonResultFail(req.status);}, 0);};req.send(null);}function onResultFail(err) {var overlay = new tr.ui.b.Overlay();overlay.textContent = err + ': ' + url + ' could not be loaded';overlay.title = 'Failed to fetch data';overlay.visible = true;}function onResult(result) {model = new tr.Model();var i = new tr.importer.Import(model);var p = i.importTracesWithProgressDialog([result]);p.then(onModelLoaded, onImportFail);}function onModelLoaded() {viewer.model = model;var urlParams = new URLSearchParams(window.location.search);var title = urlParams.get("title") || "";document.getElementById('title').innerHTML = title;}function onImportFail(err) {var overlay = new tr.ui.b.Overlay();overlay.textContent = tr.b.normalizeException(err).message;overlay.title = 'Import error';overlay.visible = true;}document.addEventListener('WebComponentsReady', function () {var container = document.createElement('track-view-container');container.id = 'track_view_container';viewer = document.createElement('tr-ui-timeline-view');viewer.track_view_container = container;Polymer.dom(viewer).appendChild(container);viewer.id = 'trace-viewer';viewer.globalMode = true;Polymer.dom(document.body).appendChild(viewer);var urlParams = new URLSearchParams(window.location.search);url = urlParams.get("tracing_url");load();});}());</script>
</head>
<body>
</body>
</html>

使用方式:http://path/to/this.html?tracing_url=path/to/tracing.json

踩过的坑

JSON 文件里,如果出现 AStart-BStart-AEnd-BEnd 这样的交叉式多行数据,有可能会导致解析失败。解决方法就是拆解为正常的顺序 AStart-AEnd-BStart-BEnd

还有,如果发现某个 Event 的 Start 和 End 明明在那里,但却还是报找不到 End 的话,通常是这个 End 被其他 Start 匹配了(即使名字不一样),所以 root cause 很可能是因为那个 solo Start。

小结

Trace Viewer 是一款强大的可视化利器,虽然通常被拿来做性能分析,但它的使用场景其实很广,至于能做出什么来,就看你的想象力咯。

这篇关于强大的可视化利器 Chrome Trace Viewer 使用详解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1032723

相关文章

C语言中联合体union的使用

本文编辑整理自: http://bbs.chinaunix.net/forum.php?mod=viewthread&tid=179471 一、前言 “联合体”(union)与“结构体”(struct)有一些相似之处。但两者有本质上的不同。在结构体中,各成员有各自的内存空间, 一个结构变量的总长度是各成员长度之和。而在“联合”中,各成员共享一段内存空间, 一个联合变量

Spring Cloud:构建分布式系统的利器

引言 在当今的云计算和微服务架构时代,构建高效、可靠的分布式系统成为软件开发的重要任务。Spring Cloud 提供了一套完整的解决方案,帮助开发者快速构建分布式系统中的一些常见模式(例如配置管理、服务发现、断路器等)。本文将探讨 Spring Cloud 的定义、核心组件、应用场景以及未来的发展趋势。 什么是 Spring Cloud Spring Cloud 是一个基于 Spring

可视化实训复习篇章

前言: 今天,我们来学习seaborn库可视化,当然,这个建立在Matplotlib的基础上,话不多说,进入今天的正题吧!当然,这个是《python数据分析与应用》书中,大家有需求的可以参考这本书。 知识点: Matplotlib中有两套接口分别是pyplot和pyylab,即绘图时候主要导入的是Matplotlib库下的两个子模块(两个py文件)matplotlib.pyplot和matp

Tolua使用笔记(上)

目录   1.准备工作 2.运行例子 01.HelloWorld:在C#中,创建和销毁Lua虚拟机 和 简单调用。 02.ScriptsFromFile:在C#中,对一个lua文件的执行调用 03.CallLuaFunction:在C#中,对lua函数的操作 04.AccessingLuaVariables:在C#中,对lua变量的操作 05.LuaCoroutine:在Lua中,

Vim使用基础篇

本文内容大部分来自 vimtutor,自带的教程的总结。在终端输入vimtutor 即可进入教程。 先总结一下,然后再分别介绍正常模式,插入模式,和可视模式三种模式下的命令。 目录 看完以后的汇总 1.正常模式(Normal模式) 1.移动光标 2.删除 3.【:】输入符 4.撤销 5.替换 6.重复命令【. ; ,】 7.复制粘贴 8.缩进 2.插入模式 INSERT

Lipowerline5.0 雷达电力应用软件下载使用

1.配网数据处理分析 针对配网线路点云数据,优化了分类算法,支持杆塔、导线、交跨线、建筑物、地面点和其他线路的自动分类;一键生成危险点报告和交跨报告;还能生成点云数据采集航线和自主巡检航线。 获取软件安装包联系邮箱:2895356150@qq.com,资源源于网络,本介绍用于学习使用,如有侵权请您联系删除! 2.新增快速版,简洁易上手 支持快速版和专业版切换使用,快速版界面简洁,保留主

如何免费的去使用connectedpapers?

免费使用connectedpapers 1. 打开谷歌浏览器2. 按住ctrl+shift+N,进入无痕模式3. 不需要登录(也就是访客模式)4. 两次用完,关闭无痕模式(继续重复步骤 2 - 4) 1. 打开谷歌浏览器 2. 按住ctrl+shift+N,进入无痕模式 输入网址:https://www.connectedpapers.com/ 3. 不需要登录(也就是

十四、观察者模式与访问者模式详解

21.观察者模式 21.1.课程目标 1、 掌握观察者模式和访问者模式的应用场景。 2、 掌握观察者模式在具体业务场景中的应用。 3、 了解访问者模式的双分派。 4、 观察者模式和访问者模式的优、缺点。 21.2.内容定位 1、 有 Swing开发经验的人群更容易理解观察者模式。 2、 访问者模式被称为最复杂的设计模式。 21.3.观察者模式 观 察 者 模 式 ( Obser

【操作系统】信号Signal超详解|捕捉函数

🔥博客主页: 我要成为C++领域大神🎥系列专栏:【C++核心编程】 【计算机网络】 【Linux编程】 【操作系统】 ❤️感谢大家点赞👍收藏⭐评论✍️ 本博客致力于知识分享,与更多的人进行学习交流 ​ 如何触发信号 信号是Linux下的经典技术,一般操作系统利用信号杀死违规进程,典型进程干预手段,信号除了杀死进程外也可以挂起进程 kill -l 查看系统支持的信号

Windows/macOS/Linux 安装 Redis 和 Redis Desktop Manager 可视化工具

本文所有安装都在macOS High Sierra 10.13.4进行,Windows安装相对容易些,Linux安装与macOS类似,文中会做区分讲解 1. Redis安装 1.下载Redis https://redis.io/download 把下载的源码更名为redis-4.0.9-source,我喜欢跟maven、Tomcat放在一起,就放到/Users/zhan/Documents