Go语言:利用pprof工具查找goroutine(协程)泄漏的示例

goroutine泄漏指的是goroutine启动之后没有退出导致goroutine的数量持续上升,或者是在实际应用中goroutine占用了很长时间才退出导致在一段时间内goroutine的数量急剧上升。通常可以采用Go自带的pprof工具来定位问题,如下面这个示例:

这是一个简单的HTTP服务,当接收到请求时另起一个goroutine来输出日志,同时返回“Hello, World!\n”。在记录日志之前可能要处理一个耗时很长的业务逻辑,如通过公网请求第三方的API接口,这里为了简化问题用time.Sleep来示意。

goroutineleak.go

package main

import (
	"io"
	"log"
	"net/http"
	_ "net/http/pprof"
	"time"
)

func writeLog(msg string) {
	go func() {
		time.Sleep(1 * time.Second)
		log.Println(msg)
	}()
}
func handler(w http.ResponseWriter, r *http.Request) {
	writeLog(r.URL.RequestURI())
	io.WriteString(w, "Hello, world!\n")
}

func main() {
	http.HandleFunc("/", handler)
	http.ListenAndServe(":8080", nil)
}

 利用Grinder来压测,开8个进程,每个进程开3个线程,同时请求上述服务。在请求过程中,通过浏览器查看http://pub.pengpengzhou.com:8080/debug/pprof/goroutine?debug=1, 域名是上述服务所在的地址。

从pprof的返回结果中,我们可以看到协程总数是5605,下面列出了5组产生协程的代码stack trace,按产生的协程数量倒排序。可以看到第一组产生了5597个协程,产生位置在goroutineleak.go的13行“time.Sleep(1 * time.Second)”对应的函数是main.writeLog。问题很快得到定位,把这一行注释掉,重新再压测,可以得到如下结果, 协程总数一下就降到了8,泄漏的问题得到了解决。

当然,在实际应用中,耗时是不可避免的,通常是用设置超时的办法来规避长时间等待。

 

相关文章:

《Go语言:利用pprof工具排查内存泄漏的示例》

《Go语言:利用pprof工具查看CPU占用情况的示例》

  • 3
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值