一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

腳本之家,腳本語言編程技術及教程分享平臺!
分類導航

Python|VBS|Ruby|Lua|perl|VBA|Golang|PowerShell|Erlang|autoit|Dos|bat|

服務器之家 - 腳本之家 - Golang - Go程序性能優化及pprof使用方法詳解

Go程序性能優化及pprof使用方法詳解

2020-05-10 15:12snowInPluto Golang

這篇文章主要為大家詳細介紹了Go程序性能優化及pprof的使用方法,具有一定的參考價值,感興趣的小伙伴們可以參考一下

Go 程序的性能優化及 pprof 的使用

程序的性能優化無非就是對程序占用資源的優化。對于服務器而言,最重要的兩項資源莫過于 CPU 和內存。性能優化,就是在對于不影響程序數據處理能力的情況下,我們通常要求程序的 CPU 的內存占用盡量低。反過來說,也就是當程序 CPU 和內存占用不變的情況下,盡量地提高程序的數據處理能力或者說是吞吐量。

Go 的原生工具鏈中提供了非常多豐富的工具供開發者使用,其中包括 pprof。

對于 pprof 的使用要分成下面兩部分來說。

Web 程序使用 pprof

先寫一個簡單的 Web 服務程序。程序在 9876 端口上接收請求。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
package main
 
import (
  "bytes"
  "io/ioutil"
  "log"
  "math/rand"
  "net/http"
 
  _ "net/http/pprof"
)
 
func main() {
  http.HandleFunc("/test", handler)
  log.Fatal(http.ListenAndServe(":9876", nil))
}
 
func handler(w http.ResponseWriter, r *http.Request) {
  err := r.ParseForm()
  if nil != err {
    w.Write([]byte(err.Error()))
    return
  }
  doSomeThingOne(10000)
  buff := genSomeBytes()
  b, err := ioutil.ReadAll(buff)
  if nil != err {
    w.Write([]byte(err.Error()))
    return
  }
  w.Write(b)
}
 
func doSomeThingOne(times int) {
  for i := 0; i < times; i++ {
    for j := 0; j < times; j++ {
 
    }
  }
}
 
func genSomeBytes() *bytes.Buffer {
  var buff bytes.Buffer
  for i := 1; i < 20000; i++ {
    buff.Write([]byte{'0' + byte(rand.Intn(10))})
  }
  return &buff
}

可以看到我們只是簡單地引入了 net/http/pprof ,并未顯示地使用。

啟動程序。

我們用 wrk 來簡單地模擬請求。

wrk -c 400 -t 8 -d 3m http://localhost:9876/test

這時我們打開 http://localhost:9876/debug/pprof,會顯示如下頁面:

Go程序性能優化及pprof使用方法詳解

用戶可以點擊相應的鏈接瀏覽內容。不過這不是我們重點講述的,而且這些內容看起來并不直觀。

我們打開鏈接 http://localhost:9876/debug/pprof/profile 稍后片刻,可以下載到文件 profile。

使用 Go 自帶的 pprof 工具打開。go tool pprof test profile。(proof 后跟的 test 為程序編譯的可執行文件)

輸入 top 命令得到:

Go程序性能優化及pprof使用方法詳解

可以看到 cpu 占用前 10 的函數,我們可以對此分析進行優化。

只是這樣可能還不是很直觀。

我們輸入命令 web(需要事先安裝 graphviz,macOS 下可以 brew install graphviz),會在瀏覽器中打開界面如下:

Go程序性能優化及pprof使用方法詳解

可以看到 main.doSomeThingOne 占用了 92.46% 的 CPU 時間,需要對其進行優化。

Web 形式的 CPU 時間圖對于優化已經完全夠用,這邊再介紹一下火焰圖的生成。macOS 推薦使用 go-torch 工具。使用方法和 go tool pprof 相似。

go-torch test profile 會生成 torch.svg 文件。可以用瀏覽器打開,如圖。

Go程序性能優化及pprof使用方法詳解

剛才只是講了 CPU 的占用分析文件的生成查看,其實內存快照的生成相似。http://localhost:9876/debug/pprof/heap,會下載得到 heap.gz 文件。

我們同樣可以使用 go tool pprof test heap.gz,然后輸入 top web 命令查看相關內容。

Go程序性能優化及pprof使用方法詳解

Go程序性能優化及pprof使用方法詳解

通用程序使用 pprof

我們寫的 Go 程序并非都是 Web 程序,這時候再使用上面的方法就不行了。

我們仍然可以使用 pprof 工具,但引入的位置為 runtime/pprof

這里貼出兩個函數,作為示例:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
// 生成 CPU 報告
func cpuProfile() {
  f, err := os.OpenFile("cpu.prof", os.O_RDWR|os.O_CREATE, 0644)
  if err != nil {
    log.Fatal(err)
  }
  defer f.Close()
 
  log.Println("CPU Profile started")
  pprof.StartCPUProfile(f)
  defer pprof.StopCPUProfile()
 
  time.Sleep(60 * time.Second)
  fmt.Println("CPU Profile stopped")
}
 
// 生成堆內存報告
func heapProfile() {
  f, err := os.OpenFile("heap.prof", os.O_RDWR|os.O_CREATE, 0644)
  if err != nil {
    log.Fatal(err)
  }
  defer f.Close()
 
  time.Sleep(30 * time.Second)
 
  pprof.WriteHeapProfile(f)
  fmt.Println("Heap Profile generated")
}

兩個函數分別會生成 cpu.profheap.prof 文件。仍然可以使用 go tool pprof 工具進行分析,在此就不贅述。

Trace 報告

直接貼代碼:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
// 生成追蹤報告
func traceProfile() {
  f, err := os.OpenFile("trace.out", os.O_RDWR|os.O_CREATE, 0644)
  if err != nil {
    log.Fatal(err)
  }
  defer f.Close()
 
  log.Println("Trace started")
  trace.Start(f)
  defer trace.Stop()
 
  time.Sleep(60 * time.Second)
  fmt.Println("Trace stopped")
}

使用工具 go tool trace 進行分析,會得到非常詳細的追蹤報告,供更深入的程序分析優化。由于報告內容比較復雜,且使用方法類似,就不繼續了。讀者可自行嘗試。

貼張網上的圖給大家大概看一下:

Go程序性能優化及pprof使用方法詳解

參考:https://github.com/caibirdme/hand-to-hand-optimize-go

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持服務器之家。

原文鏈接:http://www.cnblogs.com/snowInPluto/p/7403097.html

延伸 · 閱讀

精彩推薦
  • Golanggolang如何使用struct的tag屬性的詳細介紹

    golang如何使用struct的tag屬性的詳細介紹

    這篇文章主要介紹了golang如何使用struct的tag屬性的詳細介紹,從例子說起,小編覺得挺不錯的,現在分享給大家,也給大家做個參考。一起跟隨小編過來看...

    Go語言中文網11352020-05-21
  • Golanggolang json.Marshal 特殊html字符被轉義的解決方法

    golang json.Marshal 特殊html字符被轉義的解決方法

    今天小編就為大家分享一篇golang json.Marshal 特殊html字符被轉義的解決方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧 ...

    李浩的life12792020-05-27
  • GolangGolang中Bit數組的實現方式

    Golang中Bit數組的實現方式

    這篇文章主要介紹了Golang中Bit數組的實現方式,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧...

    天易獨尊11682021-06-09
  • GolangGolang通脈之數據類型詳情

    Golang通脈之數據類型詳情

    這篇文章主要介紹了Golang通脈之數據類型,在編程語言中標識符就是定義的具有某種意義的詞,比如變量名、常量名、函數名等等,Go語言中標識符允許由...

    4272021-11-24
  • Golanggolang 通過ssh代理連接mysql的操作

    golang 通過ssh代理連接mysql的操作

    這篇文章主要介紹了golang 通過ssh代理連接mysql的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧...

    a165861639710342021-03-08
  • Golanggolang的httpserver優雅重啟方法詳解

    golang的httpserver優雅重啟方法詳解

    這篇文章主要給大家介紹了關于golang的httpserver優雅重啟的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,...

    helight2992020-05-14
  • Golanggo日志系統logrus顯示文件和行號的操作

    go日志系統logrus顯示文件和行號的操作

    這篇文章主要介紹了go日志系統logrus顯示文件和行號的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧...

    SmallQinYan12302021-02-02
  • Golanggo語言制作端口掃描器

    go語言制作端口掃描器

    本文給大家分享的是使用go語言編寫的TCP端口掃描器,可以選擇IP范圍,掃描的端口,以及多線程,有需要的小伙伴可以參考下。 ...

    腳本之家3642020-04-25
主站蜘蛛池模板: 日本乱人伦中文在线播放 | 国产精品一区二区三区久久 | 红杏劫 | 久久99re热在线观看视频 | 冰山美人调教耻辱h | 精品欧美日韩一区二区三区 | 99只有精品 | 女人张开腿让男人桶视频免费大全 | 精品一区二区三区五区六区七区 | www.日日操 | 美国女艳星brandilove | 亚洲欧美日本在线观看 | 男女一级簧色带 | 唯美清纯 自拍偷 | 国产麻豆精品视频 | 九色PORNY丨视频入口 | 精品久久99麻豆蜜桃666 | 四虎在线网址 | 国产一区二区三区四卡 | 欧美在线视频一区 | 日韩视频免费观看 | 欧美一级视频在线 | 国产成人咱精品视频免费网站 | 亚洲 欧美 日本 国产 高清 | 亚洲国产在线播放 | 欧美肥胖老妇做爰变态 | 爽爽影院免费观看 | 亚洲一区二区三区深夜天堂 | 日韩高清成人毛片不卡 | 亚洲福利一区二区精品秒拍 | 国内自拍网红在线自拍综合 | 叛佛 作者满栀小说免费阅读 | 国产欧美va欧美va香蕉在线观 | 人人干国产 | 亚洲高清中文字幕精品不卡 | 性的张力 | japan孕妇孕交 | 久久五月综合婷婷中文云霸高清 | 日韩夫妻性生活 | 嗯好爽视频 | 被夫上司强迫中文 |