Go 服务调试精解

2024-08-29 12:36
文章标签 服务 go 调试 精解

本文主要是介绍Go 服务调试精解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

        生产环境总是会遇到一些奇怪的问题,比如 Go 服务时不时地响应非常慢甚至完全没有响应,Go 服务的内存占用总是居高不下等。遇到这些问题该如何排查与分析呢?Go 语言其实为我们提供了一些非常有用的工具,如 ppprof、Trace,这两种工具可以帮助我们分析和解决 Go 服务的性能问题。另外,学习 C 语言的读者可能知道 GDB 可以用来调试 C 程序,那么 Go 语言有没有对应的调试工具呢? Go 语言专用的调试工具叫作 dlv。

1. Go 程序分析利器 ppprof

        pprof 是 Go 语言提供的一款非常强大的性能分析工具,它可以收集 Go 程序的各项运行时指标数据,包括内存、CPU、锁等。有了这些指标数据,大部分的 Go 服务性能问题也就可以迎刃而解了。

1.1 pprof 概述 

        为什么 pprof 可以帮助我们分析 Go 程序的性能呢?因为它可以采集 Go 服务的运行时数据,比如协程调用栈、内存分配情况等。这样一来,我们就能清楚地知道 Go 服务在哪里阻塞,在哪里消耗内存。当然,要想通过 pprof 分析程序性能,需要引入一点代码,如下所示:

package main
import ("net/http"_"net/http/pprof"
)func main(){go func(){http.ListenAndServe("0.0.0.0:6060",nil)}()
}

        参考上面的官方说明,可以看到,pprof 工具可以用来分析内存溢出问题、协程溢出问题、锁/阻塞问题等。那么如何应用这些数据指标呢?举一个例子,指标 goroutine 可以采集所有协程的调用栈,通常可以用来分析 Go 服务的阻塞情况,比如当大量协程阻塞在获取锁的代码时,那是不是有可能是因为锁没有被释放?再比如当大量协程阻塞在写管道的代码时,那是不是有可能是因为读管道的协程太慢或者异常退出了?goroutine 指标的输出内容如下所示:

http://127.0.0.1:8888/debug/pprof/goroutine?debug=1//第一个数字表示协程数
1 @ 0x10390d6 0x1032357 0x1063929 0x10d0372 0x10d16da 0x10d16c8 0x1126c29 0x1135d85 0x131dc9f 0x1069921
#    0x1063928    internal/poll.runtime_pollWait+0x88        /go1.23/src/runtime/netpoll.go:302
#    0x10d0371    internal/poll.(*pollDesc).wait+0x31        /go1.23/src/internal/poll/fd_poll_runtime.go:83
#    0x10d16d9    internal/poll.(*pollDesc).waitRead+0x259    /go1.23/src/internal/poll/fd_poll_runtime.go:88
#    0x10d16c7    internal/poll.(*FD).Read+0x247            /go1.23/src/internal/poll/fd_unix.go:167
#    0x1126c28    net.(*netFD).Read+0x28                /go1.23/src/net/fd_posix.go:55
#    0x1135d84    net.(*conn).Read+0x44                 /go1.23/src/net/net.go:183
#    0x131dc9e    net/http.(*connReader).backgroundRead+0x3e    /go1.23/src/net/http/server.go:672......

        最后思考一个问题:为什么只需要引入 net/http/pprof 就能采集这些运行时数据指标呢?回顾一下上面的程序示例,我们还启动了一个HTTP服务,但是我们没有设置 HTTP 请求的处理器,那么当我们在浏览器中输入地址 “/debug/pprof" 时,该请求由谁处理了呢?其实在引入 net/http/pprof包的时候,其声明的 init 函数就默认注册好了 HTTP 请求处理器,所以我们才能通过这些接口获取到运行时数据指标,如下所示:

func init() {//前缀匹配,包含allocs、heap、goroutine等http.HandleFunc("/debug/pprof/", Index)http.HandleFunc("/debug/pprof/cmdline", Cmdline)http.HandleFunc("/debug/pprof/profile", Profile)http.HandleFunc("/debug/pprof/symbol", Symbol)http.HandleFunc("/debug/pprof/trace", Trace)
}

1.2 内存指标分析

        内存指标可以通过地址 “/debug/pprof/heap" 或者 “/debug/pprof/allocs" 查看,这两种指标采样的数据基要上是一样的,只是指标 heap 可以用来采样存活对象的内存分配情况(可通过参数 gc=1 在采样前运行 GC, 这样剩下的都是存活对象了)。以指标 heap 为例,其输出结果如下所示:

//访问地址
http://127.0.0.1:6060/debug/pprof/heap?debug=1&gc=1//heap profile 汇总数据: 
//3: 29728 [16: 54144] 含义如下:
//inuse对象数目(已分配的,去除了已释放的): inuse字节 [已分配对象数目: 已分配字节] 
//heap/1048576 平均采样频率1048576字节
heap profile: 10: 29728 [16: 54144] &

这篇关于Go 服务调试精解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Java中使用Java Mail实现邮件服务功能示例

《Java中使用JavaMail实现邮件服务功能示例》:本文主要介绍Java中使用JavaMail实现邮件服务功能的相关资料,文章还提供了一个发送邮件的示例代码,包括创建参数类、邮件类和执行结... 目录前言一、历史背景二编程、pom依赖三、API说明(一)Session (会话)(二)Message编程客

基于Go语言实现一个压测工具

《基于Go语言实现一个压测工具》这篇文章主要为大家详细介绍了基于Go语言实现一个简单的压测工具,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录整体架构通用数据处理模块Http请求响应数据处理Curl参数解析处理客户端模块Http客户端处理Grpc客户端处理Websocket客户端

Go中sync.Once源码的深度讲解

《Go中sync.Once源码的深度讲解》sync.Once是Go语言标准库中的一个同步原语,用于确保某个操作只执行一次,本文将从源码出发为大家详细介绍一下sync.Once的具体使用,x希望对大家有... 目录概念简单示例源码解读总结概念sync.Once是Go语言标准库中的一个同步原语,用于确保某个操

windos server2022的配置故障转移服务的图文教程

《windosserver2022的配置故障转移服务的图文教程》本文主要介绍了windosserver2022的配置故障转移服务的图文教程,以确保服务和应用程序的连续性和可用性,文中通过图文介绍的非... 目录准备环境:步骤故障转移群集是 Windows Server 2022 中提供的一种功能,用于在多个

Go语言实现将中文转化为拼音功能

《Go语言实现将中文转化为拼音功能》这篇文章主要为大家详细介绍了Go语言中如何实现将中文转化为拼音功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 有这么一个需求:新用户入职 创建一系列账号比较麻烦,打算通过接口传入姓名进行初始化。想把姓名转化成拼音。因为有些账号即需要中文也需要英

解决systemctl reload nginx重启Nginx服务报错:Job for nginx.service invalid问题

《解决systemctlreloadnginx重启Nginx服务报错:Jobfornginx.serviceinvalid问题》文章描述了通过`systemctlstatusnginx.se... 目录systemctl reload nginx重启Nginx服务报错:Job for nginx.javas

Go语言使用Buffer实现高性能处理字节和字符

《Go语言使用Buffer实现高性能处理字节和字符》在Go中,bytes.Buffer是一个非常高效的类型,用于处理字节数据的读写操作,本文将详细介绍一下如何使用Buffer实现高性能处理字节和... 目录1. bytes.Buffer 的基本用法1.1. 创建和初始化 Buffer1.2. 使用 Writ

C++中实现调试日志输出

《C++中实现调试日志输出》在C++编程中,调试日志对于定位问题和优化代码至关重要,本文将介绍几种常用的调试日志输出方法,并教你如何在日志中添加时间戳,希望对大家有所帮助... 目录1. 使用 #ifdef _DEBUG 宏2. 加入时间戳:精确到毫秒3.Windows 和 MFC 中的调试日志方法MFC

Go Gorm 示例详解

《GoGorm示例详解》Gorm是一款高性能的GolangORM库,便于开发人员提高效率,本文介绍了Gorm的基本概念、数据库连接、基本操作(创建表、新增记录、查询记录、修改记录、删除记录)等,本... 目录1. 概念2. 数据库连接2.1 安装依赖2.2 连接数据库3. 数据库基本操作3.1 创建表(表关

Go信号处理如何优雅地关闭你的应用

《Go信号处理如何优雅地关闭你的应用》Go中的优雅关闭机制使得在应用程序接收到终止信号时,能够进行平滑的资源清理,通过使用context来管理goroutine的生命周期,结合signal... 目录1. 什么是信号处理?2. 如何优雅地关闭 Go 应用?3. 代码实现3.1 基本的信号捕获和优雅关闭3.2