火焰图对 Go 程序进行性能分析

本文涉及的产品
模型训练 PAI-DLC,100CU*H 3个月
模型在线服务 PAI-EAS,A10/V100等 500元 1个月
交互式建模 PAI-DSW,每月250计算时 3个月
简介: 火焰图对 Go 程序进行性能分析

软件工程中,系统上线之后,仍需要持续对系统进行优化或者重构。

学会对应用系统进行运行时数据采集与性能分析是软件工程实践常用的基本技能。通常使用 profile 表示性能分析与采集,或者使用 profiling 代表性能分析这个行为。比如 Java 语言中相关的工具为 jprofiler,意为 Java Profiler。

Go 语言非常注重性能,其内置库里就自带了性能分析库 pprof。pprof 有两个包用来分析程序: runtime/pprof 与 net/http/pprof,其中 net/http/pprof 只是对 runtime/pprof 包进行封装并用 http 暴露出来。runtime/pprof 用于对普通的应用程序进行性能分析,主要用于可结束的代码块,比如一次函数调用;而 net/http/pprof 专门用于对后台服务型程序的性能采集与分析。

本小节将会介绍如何基于 pprof 进行性能分析与优化,包括 CPU 、内存占用、Block 阻塞以及 Goroutine 使用等方面。除此之外,还会介绍更加直观的图形工具:火焰图,基于 go-torch 将 pprof 的结果转换成火焰图。

普通应用程序的性能分析

我们已经知道,runtime/pprof 用于对普通的应用程序进行性能分析,主要用于可结束的代码块。因此,我们下面通过案例来实践。

计算圆周率

笔者选取的案例是计算圆周率的算法。

众所周知,可以说,它是世界上最有名的无理常数了,代表的是一个圆的周长与直径之比或称为“圆周率”。公元前 250 年左右,阿基米德给出了“圆周率”的估计值在  223/71~22/7 之间。

中国南北朝时期的著名数学家祖冲之(429-500)首次将“圆周率”精算到小数第七位,即在 3.1415926 至 3.1415927 之间,他提出的“密率与约率”对数学的研究有重大贡献。直到 15 世纪,阿拉伯数学家阿尔·卡西才以“精确到小数点后17位”打破了这一纪录。

代表“圆周率”的字母是第十六个希腊字母的小写。也是希腊语 περιφρεια(表示周边,地域,圆周)的首字母。1706 年英国数学家威廉·琼斯(William Jones, 1675-1749)最先使用“π”来表示圆周率。1736 年,瑞士数学家欧拉(Leonhard Euler, 1707-1783)也开始用表示圆周率。从此,便成了圆周率的代名词。

通常的计算方法有如下几种:

  • 蒙特卡罗法;
  • 正方形逼近;
  • 迭代法;
  • 丘德诺夫斯基公式

测试代码的实现

笔者这里采用蒙特卡罗方法计算圆周率,大致思路如下:

正方形内部有一个相切的圆,它们的面积之比是π/4。 在这个正方形内部,随机产生10000个点(即10000个坐标对 (x, y)),计算它们与中心点的距离,从而判断是否落在圆的内部。 如果这些点均匀分布,那么圆内的点应该占到所有点的 π/4,因此将这个比值乘以4,就是π的值。通过随机模拟30000个点,π的估算值与真实值相差0.07%。

最后,实现的完整代码如下所示:

package main

import (
"flag"
"fmt"
"log"
"os"
"runtime"
"runtime/pprof"
"time"
)

var n int64 = 10000000000
var h float64 = 1.0 / float64(n)

func f(a float64) float64 {
return 4.0 / (1.0 + a*a)
}

func chunk(start, end int64, c chan float64) {
var sum float64 = 0.0
for i := start; i < end; i++ {
 x := h * (float64(i) + 0.5)
 sum += f(x)
}
c <- sum * h
}

func main() {
var cpuProfile = flag.String("cpuprofile", "", "write cpu profile to file")
var memProfile = flag.String("memprofile", "", "write mem profile to file")
flag.Parse()
//采样cpu运行状态
if *cpuProfile != "" {
 f, err := os.Create(*cpuProfile)
 if err != nil {
  log.Fatal(err)
 }
 pprof.StartCPUProfile(f)
 defer pprof.StopCPUProfile()
}
//记录开始时间
start := time.Now()

var pi float64
np := runtime.NumCPU()
runtime.GOMAXPROCS(np)
c := make(chan float64, np)
fmt.Println("np: ", np)

for i := 0; i < np; i++ {
   //利用多处理器,并发处理
 go chunk(int64(i)*n/int64(np), (int64(i)+1)*n/int64(np), c)
}

for i := 0; i < np; i++ {
 tmp := <-c
 fmt.Println("c->: ", tmp)

 pi += tmp
 fmt.Println("pai: ", pi)

}

fmt.Println("Pi: ", pi)

//记录结束时间
end := time.Now()

//输出执行时间,单位为毫秒。
fmt.Printf("spend time: %vs\n", end.Sub(start).Seconds())
//采样 memory 状态
if *memProfile != "" {
 f, err := os.Create(*memProfile)
 if err != nil {
  log.Fatal(err)
 }
 pprof.WriteHeapProfile(f)
 f.Close()
}
}

如上就是计算 π 的算法,基于 go 语言的 goroutine和 channel,充分利用多核处理器,提高 CPU 资源计算的速度。

我们在依赖中引入了 runtime/pprof,在实现的代码中添加了相关的 CPU Profiling 和 Memory Profiling 代码就可以实现 CPU 和内存的性能评测。

编译与执行

接着就是编译获得可执行文件,执行后获得 pprof 的采样数据,然后就可以利用相关工具进行分析。相关的命令如下:

$ go build  -o pai main.go
$ ./pai --cpuprofile=cpu.pprof
$ ./pai --memprofile=mem.pprof


image.png

上面的命令依次生成了 cpu.pprof 和 mem.pprof 两个采样文件,我们使用 go tool pprof  命令进行分析:

$ go tool pprof cpu.pprof

执行完上述命令即进入 pprof 命令行交互模式。pprof 支持多个指令,比如 top 用于显示 pprof 文件中的前 10 项数据,可以通过top 20等方式显示20行数据;其他的指令如 list,pdf、eog 等。


上图中,其他的一些参数解释如下:

  • Duration:程序执行时间。多核执行程序,总计耗时 13.47s,而采样时间为 24.44s;每个核均分采样时间。
  • flat/flat%:分别表示在当前层级 CPU 的占用时间和百分比。
  • cum/cum%:分别表示截止到当前层级累积的 CPU 时间和占比。
  • sum%:所有层级的 CPU 时间累积占用,从小到大一直累积到100%,即 24.44s。

本例中,main.chunk 在当前层级占用 CPU 时间 21.86s,占比本次采集时间的 89.44%。而该函数累积占用时间 24.44s,占本次采集时间的 100%。通过 cum 数据可以看到,chunk 函数的 CPU 占用时间最多。

上图很清楚的说明了应用程序耗时的主要函数,接着就利用 list 命令查看占用的主要因素。list 命令根据你的正则表达式输出相关的方法,直接跟可选项 -o 会输出所有的方法,也可以指定方法名。这样就能查看匹配函数的代码以及每行代码的耗时:

image.png

从上图可以看出,在第 24 行调用函数 f(x) 还额外花了 2.58s,每一行代码花费的时间都有显示出来,根据这些信息可以开展代码的优化。

图形化渲染

对于 pprof 采集的结果,我们不仅可以使用 pprof 自带的命令进行分析,还可以通过更加直观的矢量图进行分析。借助于 graphviz,pprof 可以直接生成对应的图像化文件。

笔试基于 Centos 7.5 系统,通过如下的命令直接安装 graphviz:

$ sudo yum install graphviz

更多系统环境的安装说明,请参见 graphviz 官网

安装好 graphviz,继续在 pprof 交互命令行中输入 svg:

image.png

注意 web 命令在服务器类型的系统不支持,通过 svg 命令来生成矢量图,使用浏览器打开,如下所示:

image.png

笔者截取了部分内容,从上图同样可以看到,主要耗时的函数为 main.chunk,耗时时间为 21.86s,关联调用的函数 f(x) 耗时为 2.58s。图中各个方块的大小也代表 CPU 占用的情况,方块越大说明占用 CPU 时间越长。

后台服务程序的性能分析

针对一直运行的后台服务,比如 web 应用或者分布式应用,我们可以使用 net/http/pprof 库,它能够在应用提供 HTTP 服务时进行分析。

pprof 采集后台服务,如果使用了默认的 http.DefaultServeMux,通常是代码直接使用 http.ListenAndServe("0.0.0.0:8000", nil),这种情况则比较简单,只需要导入包即可。

import (
_ "net/http/pprof"
)

注意该包利用下划线"_"导入,意味着我们只需要该包运行其init()函数即可,如此该包将自动完成信息采集并保存在内存中。

如果你使用自定义的 ServerMux复用器,则需要手动注册一些路由规则:

r.HandleFunc("/debug/pprof/", pprof.Index)
r.HandleFunc("/debug/pprof/heap", pprof.Index)
r.HandleFunc("/debug/pprof/cmdline", pprof.Cmdline)
r.HandleFunc("/debug/pprof/profile", pprof.Profile)
r.HandleFunc("/debug/pprof/symbol", pprof.Symbol)
r.HandleFunc("/debug/pprof/trace", pprof.Trace)

这些路径分别表示:

  • /debug/pprof/profile:访问这个链接会自动进行 CPU profiling,持续 30s,并生成一个文件供下载,可以通过带参数?=seconds=60进行60秒的数据采集。
  • /debug/pprof/block:Goroutine阻塞事件的记录。默认每发生一次阻塞事件时取样一次。
  • /debug/pprof/goroutines:活跃Goroutine的信息的记录。仅在获取时取样一次。
  • /debug/pprof/heap: 堆内存分配情况的记录。默认每分配512K字节时取样一次。
  • /debug/pprof/mutex: 查看争用互斥锁的持有者。
  • /debug/pprof/threadcreate: 系统线程创建情况的记录。 仅在获取时取样一次。

改写测试代码

将计算圆周率的程序改写成一个服务,对外提供一个接口,并引入 net/http/pprof 依赖,来采集 HTTP 服务的性能指标。

package main

import (
"fmt"
"net/http"
_ "net/http/pprof"
"runtime"
)

var n int64 = 10000000000
var h = 1.0 / float64(n)

funcf(a float64)float64 {
return4.0 / (1.0 + a*a)
}

funcchunk(start, end int64, c chanfloat64) {
var sum float64 = 0.0
for i := start; i < end; i++ {
 x := h * (float64(i) + 0.5)
 sum += f(x)
}
c <- sum * h
}

funccallFunc(w http.ResponseWriter, r *http.Request) {

var pi float64
np := runtime.NumCPU()
runtime.GOMAXPROCS(np)
c := make(chanfloat64, np)
fmt.Println("np: ", np)

for i := 0; i < np; i++ {
 go chunk(int64(i)*n/int64(np), (int64(i)+1)*n/int64(np), c)
}

for i := 0; i < np; i++ {
 tmp := <-c
 fmt.Println("c->: ", tmp)

 pi += tmp
 fmt.Println("pai: ", pi)

}

fmt.Println("Pi: ", pi)
}

funcmain() {
http.HandleFunc("/getAPi", callFunc)
http.ListenAndServe(":8000", nil)
}

我们在上述代码的实现中,对外暴露了 8000 端口,并定义了一个接口 getAPi。计算圆周率的实现和之前相同,每次调用接口都将会触发计算 π 一次。

编译执行

该写完代码,我们就可以进行编译和执行 HTTP 服务了,执行如下的命令:

$ go build -o httpapi main.go

$ ./httpapi

将程序编译成功之后,运行二进制文件,可以获取服务的性能数据后,

此时,我们就可以通过 pprof 的 HTTP 接口访问 http://localhost:8000/debug/pprof/:

image.png

上图展示了 pprof web 查看服务的运行情况,同时不断刷新网页可以发现采样结果也在不断更新。

图形化分析

与上面可结束的程序进行性能分析一样,我们对于后台程序也可以使用图像化的方式分析性能。

接下来使用 go tool pprof 工具对这些数据进行分析和保存了,一般都是使用 pprof 通过 HTTP 访问上面列的那些路由端点直接获取到数据后再进行分析,获取到数据后 pprof 会自动让终端进入交互模式。

通过如下的命令查看内存 Memory 相关情况:

$ go tool pprof main http://localhost:8000/debug/pprof/heap

image.png

上述命令采集内存信息,控制台输出了生成的图片名称:profile001.svg,默认在当前目录,当然我们也可以指定位置和文件名。

image.png

由于没有 http 请求的访问,因此内存的占用比较低,没有任何异常。下面我们将通过压测模拟线上情况,来分析在正常运行时的各项性能。

利用 go-torch 生成火焰图

上面的小节介绍了 net/http/pprof 和 runtime/pprof 进行 Go 程序的性能分析。然而上面的案例仅仅只是采样了部分代码段。同时只有当有大量请求时才能看到应用服务的主要优化信息。这时候就需要借助于另一款 Uber 开源的火焰图工具 go-torch,以便辅助我们完成分析。要想实现火焰图的效果,需要安装如下 3 个工具:压测组件 wrk、FlameGraph 火焰图、go-torch 工具。下面将会依次介绍这三款组件的安装使用。

压测组件 wrk

wrk 是一款针对 HTTP 协议的基准测试工具,它能够在单机多核 CPU 的条件下,使用系统自带的高性能 I/O 机制,如 epoll,kqueue 等,通过多线程和事件模式,对目标机器产生大量的负载。安装命令如下所示:

$ git clone https://github.com/brendangregg/FlameGraph.git
$ cd wrk/
$ make

通过如上的命令,我们就生成了可执行的 wrk 文件。其使用比较简单,主要参数说明如下:

  • -c:总的连接数(每个线程处理的连接数=总连接数/线程数)
  • -d:测试的持续时间,如2s(2second),2m(2minute),2h(hour)
  • -t:需要执行的线程总数
  • -s:执行Lua脚本,这里写lua脚本的路径和名称,后面会给出案例
  • -H:需要添加的头信息,注意header的语法,举例,-H “token: abcdef”,说明一下,token,冒号,空格,abcdefg(不要忘记空格,否则会报错的)。

笔者刚开始执行的压测参数如下:

./wrk -t5 -c10 -d120s http://localhost:8000/getAPi

即 5 个线程并发,每秒保持 10 个连接,持续时间 120s。如果出现如下的错误,

unable to create thread 419: Too many open files

这是由于 /socket连接数量超过系统设定值,则需要调整每个用户最大允许打开文件数量。

$ ulimit -n 2048

FlameGraph 火焰图与 go-torch

火焰图(flame graph)是性能分析的利器,通过它可以快速定位性能瓶颈点。在 Linux 服务器,一般配合 perf 一起使用。

go-torch 是 uber 开源的一个工具,可以直接读取 pprof的 profiling 数据,并生成一个火焰图的 svg 文件。火焰图 svg 文件可以通过浏览器打开,它对于调用图的优点是:可以通过点击每个方块来分析它上面的内容。

执行如下的命令进行安装:

$ git clone https://github.com/brendangregg/FlameGraph.git
$ go get github.com/uber/go-torch

go-torch 使用的命令如下:

$ go-torch -u http://localhost:8000 -t 100

如上的命令将会开启 go-torch 工具对 http://localhost:8000 采集 100s 信息。

压测生成火焰图

安装好上述三个组件之后,我们将会进行测试。首先是启动我们的应用服务:

$ ./httpapi

接着启动压测和 go-torch:

$ ./wrk -t5 -c10 -d120s http://localhost:8000/getAPi
$ go-torch -u http://localhost:8000 -t 100

image.png

image.png

可以看到,我们压测的请求,已经在服务端生成相应的火焰图:torch.svg。注:在 FlameGraph 目录下执行 go-torch,否则需将该二进制可执行文件的路径添加到系统环境变量。

打开火焰图,如下所示:

image.png

火焰图形似火焰,故此得名,其横轴是 CPU 占用时间,纵轴是调用顺序。火焰图的调用顺序从下到上,每个方块代表一个函数,它上面一层表示这个函数会调用哪些函数,方块的大小代表了占用 CPU 使用的长短。火焰图的配色并没有特殊的意义,默认的红、黄配色是为了更像火焰而已。

与我们上面所分析的结果是一样的,总体的耗时都在 chunk 函数。我们再来看一张没有请求访问时的火焰图:

image.png

可以看到,这种情况 CPU 占用时间和内存占用非常平稳,主要集中在提供 http 服务的库函数。

小结

本文主要介绍了如何通过 pprof 对 Go 应用程序进行性能指标的采集以及性能分析。我们通过 pprof 获取到 CPU 和内存使用的细节,更进一步可以指导哪些函数耗时,函数之间的调用链。想更细致分析,就要精确到代码级别了,看看每行代码的耗时,直接定位到出现性能问题的那行代码。


结合 Uber 开源的 go-torch 生成火焰图,从全局来查看系统运行时的内存和 CPU,以及 Goroutines 和阻塞锁等情况,熟练使用性能分析的工具,能够帮助我们更快地定位线上问题并解决问题的 bug。


通过本文的讲解,你也了解到,开启后台程序的性能分析需要有请求,而不是静态的服务,本文使用的是压测来模拟大量的请求。当然在生产环境开启 pprof 也是需要考虑性能的开销,在上线前解决问题肯定是最好的选择。

相关实践学习
使用PAI-EAS一键部署ChatGLM及LangChain应用
本场景中主要介绍如何使用模型在线服务(PAI-EAS)部署ChatGLM的AI-Web应用以及启动WebUI进行模型推理,并通过LangChain集成自己的业务数据。
机器学习概览及常见算法
机器学习(Machine Learning, ML)是人工智能的核心,专门研究计算机怎样模拟或实现人类的学习行为,以获取新的知识或技能,重新组织已有的知识结构使之不断改善自身的性能,它是使计算机具有智能的根本途径,其应用遍及人工智能的各个领域。 本课程将带你入门机器学习,掌握机器学习的概念和常用的算法。
目录
相关文章
|
2月前
|
Go API 数据库
Go 语言中常用的 ORM 框架,如 GORM、XORM 和 BeeORM,分析了它们的特点、优势及不足,并从功能特性、性能表现、易用性和社区活跃度等方面进行了比较,旨在帮助开发者根据项目需求选择合适的 ORM 框架。
本文介绍了 Go 语言中常用的 ORM 框架,如 GORM、XORM 和 BeeORM,分析了它们的特点、优势及不足,并从功能特性、性能表现、易用性和社区活跃度等方面进行了比较,旨在帮助开发者根据项目需求选择合适的 ORM 框架。
134 4
|
2月前
|
Go 数据处理 开发者
Go 语言的反射机制允许程序在运行时动态检查和操作类型信息,提供极大的灵活性和扩展性
Go 语言的反射机制允许程序在运行时动态检查和操作类型信息,提供极大的灵活性和扩展性。本文探讨了反射的基本原理、主要操作、应用场景及注意事项,并通过实例展示了反射的实际应用,帮助开发者更好地理解和使用这一强大特性。
42 2
|
2月前
|
中间件 Go API
Go语言中几种流行的Web框架,如Beego、Gin和Echo,分析了它们的特点、性能及适用场景,并讨论了如何根据项目需求、性能要求、团队经验和社区支持等因素选择最合适的框架
本文概述了Go语言中几种流行的Web框架,如Beego、Gin和Echo,分析了它们的特点、性能及适用场景,并讨论了如何根据项目需求、性能要求、团队经验和社区支持等因素选择最合适的框架。
131 1
|
4月前
|
Kubernetes Go 持续交付
一个基于Go程序的持续集成/持续部署(CI/CD)
本教程通过一个简单的Go程序示例,展示了如何使用GitHub Actions实现从代码提交到Kubernetes部署的CI/CD流程。首先创建并版本控制Go项目,接着编写Dockerfile构建镜像,再配置CI/CD流程自动化构建、推送Docker镜像及部署应用。此流程基于GitHub仓库,适用于快速迭代开发。
89 3
|
4月前
|
Kubernetes 持续交付 Go
创建一个基于Go程序的持续集成/持续部署(CI/CD)流水线
创建一个基于Go程序的持续集成/持续部署(CI/CD)流水线
|
4月前
|
缓存 安全 Java
如何利用Go语言提升微服务架构的性能
在当今的软件开发中,微服务架构逐渐成为主流选择,它通过将应用程序拆分为多个小服务来提升灵活性和可维护性。然而,如何确保这些微服务高效且稳定地运行是一个关键问题。Go语言,以其高效的并发处理能力和简洁的语法,成为解决这一问题的理想工具。本文将探讨如何通过Go语言优化微服务架构的性能,包括高效的并发编程、内存管理技巧以及如何利用Go生态系统中的工具来提升服务的响应速度和资源利用率。
|
4月前
|
IDE Go 数据处理
Go to Learn Go之第一个Go程序
Go to Learn Go之第一个Go程序
33 0
|
5月前
|
存储 缓存 监控
Linux性能分析工具-perf并生成火焰图
Linux性能分析工具-perf并生成火焰图
|
5月前
|
Linux Shell Go
如何构建和安装 Go 程序
如何构建和安装 Go 程序
56 1
|
5月前
|
Go
在Go中如何停止程序
在Go中如何停止程序