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

相关文章

JavaScript中的高级调试方法全攻略指南

《JavaScript中的高级调试方法全攻略指南》什么是高级JavaScript调试技巧,它比console.log有何优势,如何使用断点调试定位问题,通过本文,我们将深入解答这些问题,带您从理论到实... 目录观点与案例结合观点1观点2观点3观点4观点5高级调试技巧详解实战案例断点调试:定位变量错误性能分

sysmain服务可以禁用吗? 电脑sysmain服务关闭后的影响与操作指南

《sysmain服务可以禁用吗?电脑sysmain服务关闭后的影响与操作指南》在Windows系统中,SysMain服务(原名Superfetch)作为一个旨在提升系统性能的关键组件,一直备受用户关... 在使用 Windows 系统时,有时候真有点像在「开盲盒」。全新安装系统后的「默认设置」,往往并不尽编

Python 基于http.server模块实现简单http服务的代码举例

《Python基于http.server模块实现简单http服务的代码举例》Pythonhttp.server模块通过继承BaseHTTPRequestHandler处理HTTP请求,使用Threa... 目录测试环境代码实现相关介绍模块简介类及相关函数简介参考链接测试环境win11专业版python

GO语言短变量声明的实现示例

《GO语言短变量声明的实现示例》在Go语言中,短变量声明是一种简洁的变量声明方式,使用:=运算符,可以自动推断变量类型,下面就来具体介绍一下如何使用,感兴趣的可以了解一下... 目录基本语法功能特点与var的区别适用场景注意事项基本语法variableName := value功能特点1、自动类型推

GO语言中函数命名返回值的使用

《GO语言中函数命名返回值的使用》在Go语言中,函数可以为其返回值指定名称,这被称为命名返回值或命名返回参数,这种特性可以使代码更清晰,特别是在返回多个值时,感兴趣的可以了解一下... 目录基本语法函数命名返回特点代码示例命名特点基本语法func functionName(parameters) (nam

Go之errors.New和fmt.Errorf 的区别小结

《Go之errors.New和fmt.Errorf的区别小结》本文主要介绍了Go之errors.New和fmt.Errorf的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考... 目录error的基本用法1. 获取错误信息2. 在条件判断中使用基本区别1.函数签名2.使用场景详细对

Nginx中配置使用非默认80端口进行服务的完整指南

《Nginx中配置使用非默认80端口进行服务的完整指南》在实际生产环境中,我们经常需要将Nginx配置在其他端口上运行,本文将详细介绍如何在Nginx中配置使用非默认端口进行服务,希望对大家有所帮助... 目录一、为什么需要使用非默认端口二、配置Nginx使用非默认端口的基本方法2.1 修改listen指令

SysMain服务可以关吗? 解决SysMain服务导致的高CPU使用率问题

《SysMain服务可以关吗?解决SysMain服务导致的高CPU使用率问题》SysMain服务是超级预读取,该服务会记录您打开应用程序的模式,并预先将它们加载到内存中以节省时间,但它可能占用大量... 在使用电脑的过程中,CPU使用率居高不下是许多用户都遇到过的问题,其中名为SysMain的服务往往是罪魁

Go语言连接MySQL数据库执行基本的增删改查

《Go语言连接MySQL数据库执行基本的增删改查》在后端开发中,MySQL是最常用的关系型数据库之一,本文主要为大家详细介绍了如何使用Go连接MySQL数据库并执行基本的增删改查吧... 目录Go语言连接mysql数据库准备工作安装 MySQL 驱动代码实现运行结果注意事项Go语言执行基本的增删改查准备工作

Go中select多路复用的实现示例

《Go中select多路复用的实现示例》Go的select用于多通道通信,实现多路复用,支持随机选择、超时控制及非阻塞操作,建议合理使用以避免协程泄漏和死循环,感兴趣的可以了解一下... 目录一、什么是select基本语法:二、select 使用示例示例1:监听多个通道输入三、select的特性四、使用se