Go語言上手(三) | 青訓營筆記

来源:https://www.cnblogs.com/WenTesla/archive/2023/04/23/17346473.html
-Advertisement-
Play Games

高質量編程簡介及編碼規範 高質量: 各種邊界條件考慮完備 異常情況處理,穩定性 易讀易維護 編程原則 簡單性 可讀性 生產力 編碼規範 公共符號始終要註釋 例外:實現介面的方法不需要註釋 格式化 使用gofmt(官方工具)自動格式化 註釋 代碼作用(適合公共符號) 代碼如何實現 (適合註釋實現過程) ...


高質量編程簡介及編碼規範

高質量:

  • 各種邊界條件考慮完備

  • 異常情況處理,穩定性

  • 易讀易維護

編程原則

  • 簡單性

  • 可讀性

  • 生產力

編碼規範

公共符號始終要註釋

例外:實現介面的方法不需要註釋

格式化

使用gofmt(官方工具)自動格式化

註釋

  • 代碼作用(適合公共符號)

  • 代碼如何實現 (適合註釋實現過程)

  • 代碼實現的原因(適合解釋代碼的外部因素和提供額外的上下文)

  • 代碼什麼情況下出錯(適合代碼的限制條件)

公共符號始終要註釋·包中聲明的每個公共的符號: 變數、常量、函數以及結構都需要添加註釋 .任何既不明顯也不簡短的公 共功能必須予以註釋 無論長度或複雜程度如何, 對庫中的任何函數都必須進行註釋

命名規範

變數

縮略詞全大寫,但當其位於變數開頭且不需要導出時,使用全小寫

  • 例如使用ServeHTTP而不是ServeHttp

  • 使用XMLHTTPRequest或者xmlHTTPRequest

  • 變數距離其被使用的地方越遠,則需要攜帶越多的上下文信息

  • 全局變數在其名字中需要更多的上下文信息,使得在不同地方可以輕易辨認出其含義

函數

函數名不攜帶包名的上下文信息,因為包名和函數名總是成對出現的·函數名儘量簡短 當名為foo的包某個函數返回類型Foo時,可以省略類型信息而不導致歧義 當名為foo的包某個函數返回類型T時(T並不是Foo),可以在函數名中加入類型信息

package

  • 只由小寫字母組成。不包含大寫字母和下劃線等字元·

  • 簡短並包含一定的上下文信息。例如schema、task 等·

  • 不要與標準庫同名。例如不要使用sync或者strings

以下規則儘量滿足,以標準庫包名為例

  • 不使用常用變數名作為包名。例如使用bufio而不是buf·使用單數而不是複數。例如使用encoding而不是encodings

  • 謹慎地使用縮寫。例如使用fmt 在不破壞上下文的情況下比 format 更加簡短

控制流程

避免嵌套

儘量保存為最小縮進

錯誤處理

簡單錯誤

  • 簡單的錯誤指的是僅出現一次的錯誤,且在其他地方不需要捕獲該錯誤

  • 優先使用errors.New來創建匿名變數來直接表示簡單錯誤

  • 如果有格式化的需求,使用fmt.Errorf

錯誤的Wrap和 Unwrap

·錯誤的Wrap 實際上是提供了一個error嵌套另一個error的能力,從而生成一個error的跟蹤鏈 ·在 fmt.Errorf中使用:%w關鍵字來將一個錯誤關聯至 錯誤鏈中

錯誤判定

  • 判定一個錯誤是否為特定錯誤,使用errors.Is

  • 不同於使用==,使用該方法可以判定錯誤鏈上的所有錯誤是否含有特定的錯誤

panic

  • 不建議在業務代碼中使用panic

  • ·調用函數不包含recover會造成程式崩潰·若問題可以被屏蔽或解決,建議使用error代替panic

  • 當程式啟動階段發生不可逆轉的錯誤時,可以在init 或 main函數中使用panic

性能優化

benchmark工具

 

slice

提前指定大小

在大切片上創建小切片,使用copy代替

string

使用strings.builder 和java類似

空結構體

使用空結構體struct{}實列不占用空間

map

map 預分配記憶體分析

  • 不斷向map中添加元素的操作會觸發map的擴容·

  • 提前分配好空間可以減少記憶體拷貝和Rehash 的消耗·

  • 建議根據實際需求提前預估好需要的空間

使用atomic 包

鎖的實現是通過操作系統來實現,屬於系統調用.atomic 操作是通過硬體實現,

效率比鎖高sync.Mutex應該用來保護一段邏輯,不僅僅用於保護一個變數。

對於非數值操作,可以使用atomic.Value,能承載一個interface}

實戰

直接拉取倉庫

wolfogre/go-pprof-practice: go pprof practice. (github.com)

分析的博客:

golang pprof 實戰 | Wolfogre's Blog

性能分析工具 pprof

項目目錄

 

沒有外部依賴,直接運行即可

保持程式運行,打開瀏覽器訪問 http://localhost:6060/debug/pprof/,可以看到如下頁面:

 

頁面上展示了可用的程式運行採樣數據,分別有:

類型描述備註
allocs 記憶體分配情況的採樣信息 可以用瀏覽器打開,但可讀性不高
blocks 阻塞操作情況的採樣信息 可以用瀏覽器打開,但可讀性不高
cmdline 顯示程式啟動命令及參數 可以用瀏覽器打開,這裡會顯示 ./go-pprof-practice
goroutine 當前所有協程的堆棧信息 可以用瀏覽器打開,但可讀性不高
heap 堆上記憶體使用情況的採樣信息 可以用瀏覽器打開,但可讀性不高
mutex 鎖爭用情況的採樣信息 可以用瀏覽器打開,但可讀性不高
profile CPU 占用情況的採樣信息 瀏覽器打開會下載文件
threadcreate 系統線程創建情況的採樣信息 可以用瀏覽器打開,但可讀性不高
trace 程式運行跟蹤信息 瀏覽器打開會下載文件,本文不涉及,可另行參閱《深入淺出 Go trace》

 

命令行

go tool pprof http://localhost:6060/debug/pprof/profile

topN 查看占用最多的函數

(pprof) top
Showing nodes accounting for 8.91s, 98.67% of 9.03s total
Dropped 35 nodes (cum <= 0.05s)
    flat flat%   sum%       cum   cum%
    8.91s 98.67% 98.67%      8.93s 98.89% github.com/wolfogre/go-pprof-practice/animal/felidae/tiger.(*Tiger).Eat
        0     0% 98.67%      8.93s 98.89% github.com/wolfogre/go-pprof-practice/animal/felidae/tiger.(*Tiger).Live
        0     0% 98.67%      8.97s 99.34% main.main
        0     0% 98.67%      8.97s 99.34% runtime.main
        0     0% 98.67%      0.05s  0.55% runtime.systemstack

flat=Cum 函數中沒有調用其他函數

flat=0 函數中只有其他函數的調用

輸入 list Eat,查看問題具體在代碼的哪一個位置:根據指定的正則表達式查找

(pprof) list Eat 
Total: 9.03s
ROUTINE ======================== github.com/wolfogre/go-pprof-practice/animal/canidae/dog.(*Dog).Eat in H:\go-pprof-practice\animal\canidae\dog\dog.go
        0       10ms (flat, cum)  0.11% of Total
        .         .     26:   d.Pee()
        .         .     27:   d.Run()
        .         .     28:   d.Howl()
        .         .     29:}
        .         .     30:func (d *Dog) Eat() {
        .       10ms     31:   log.Println(d.Name(), "eat")
        .         .     32:}
        .         .     33:
        .         .     34:func (d *Dog) Drink() {
        .         .     35:   log.Println(d.Name(), "drink")
        .         .     36:}
ROUTINE ======================== github.com/wolfogre/go-pprof-practice/animal/felidae/tiger.(*Tiger).Eat in H:\go-pprof-practice\animal\felidae\tiger\tiger.go
    8.91s      8.93s (flat, cum) 98.89% of Total
        .         .     26:無效的迴圈
        .         .     27:*/
        .         .     28:func (t *Tiger) Eat() {
        .         .     29:   log.Println(t.Name(), "eat")
        .         .     30:   loop := 10000000000
    8.91s      8.93s     31:   for i := 0; i < loop; i++ {
        .         .     32:           // do nothing
        .         .     33:   }
        .         .     34:}
        .         .     35:
        .         .     36:func (t *Tiger) Drink() {

web 調用關係可視化

可以訪問 graphviz 官網尋找適合自己操作系統的安裝方法

調查記憶體

go tool pprof http://localhost:6060/debug/pprof/heap

top

(pprof) top
Showing nodes accounting for 1.20GB, 100% of 1.20GB total
Dropped 4 nodes (cum <= 0.01GB)
    flat flat%   sum%       cum   cum%
   1.20GB   100%   100%     1.20GB   100% github.com/wolfogre/go-pprof-practice/animal/muridae/mouse.(*Mouse).Steal
        0     0%   100%     1.20GB   100% github.com/wolfogre/go-pprof-practice/animal/muridae/mouse.(*Mouse).Live
        0     0%   100%     1.20GB   100% main.main
        0     0%   100%     1.20GB   100% runtime.main

查看到占用1G多的記憶體

記憶體回收

go tool pprof http://localhost:6060/debug/pprof/allocs

top

(pprof) top
Showing nodes accounting for 592MB, 99.75% of 593.50MB total
Dropped 16 nodes (cum <= 2.97MB)
    flat flat%   sum%       cum   cum%
    592MB 99.75% 99.75%     592MB 99.75% github.com/wolfogre/go-pprof-practice/animal/canidae/dog.(*Dog).Run (inline)
        0     0% 99.75%     592MB 99.75% github.com/wolfogre/go-pprof-practice/animal/canidae/dog.(*Dog).Live
        0     0% 99.75%     592MB 99.75% main.main
        0     0% 99.75%   592.50MB 99.83% runtime.main

可以看到 github.com/wolfogre/go-pprof-practice/animal/canidae/dog.(*Dog).Run 會進行無意義的記憶體申請,而這個函數又會被頻繁調用,這才導致程式不停地進行 GC:

func (d *Dog) Run() {
log.Println(d.Name(), "run")
_ = make([]byte, 16 * constant.Mi)
}

排查協程泄露

go tool pprof http://localhost:6060/debug/pprof/goroutine
(pprof) top
Showing nodes accounting for 103, 99.04% of 104 total
Showing top 10 nodes out of 33
flat flat% sum% cum cum%
102 98.08% 98.08% 102 98.08% runtime.gopark
1 0.96% 99.04% 1 0.96% runtime.goroutineProfileWithLabels
0 0% 99.04% 100 96.15% github.com/wolfogre/go-pprof-practice/animal/canidae/wolf.(*Wolf).Drink.func1
0 0% 99.04% 1 0.96% github.com/wolfogre/go-pprof-practice/animal/felidae/cat.(*Cat).Live
0 0% 99.04% 1 0.96% github.com/wolfogre/go-pprof-practice/animal/felidae/cat.(*Cat).Pee
0 0% 99.04% 1 0.96% internal/poll.(*FD).Accept
0 0% 99.04% 1 0.96% internal/poll.(*FD).acceptOne
0 0% 99.04% 1 0.96% internal/poll.(*pollDesc).wait
0 0% 99.04% 1 0.96% internal/poll.execIO

排查鎖的爭用

go tool pprof http://localhost:6060/debug/pprof/mutex
(pprof) top
Showing nodes accounting for 126.40s, 100% of 126.40s total
flat flat% sum% cum cum%
126.40s 100% 100% 126.40s 100% sync.(*Mutex).Unlock (inline)
0 0% 100% 126.40s 100% github.com/wolfogre/go-pprof-practice/animal/canidae/wolf.(*Wolf).Howl.func1
func (w *Wolf) Howl() {
log.Println(w.Name(), "howl")

m := &sync.Mutex{}
m.Lock()
go func() {
time.Sleep(time.Second)
m.Unlock()
}()
m.Lock()
}

可以看到,這個鎖由主協程 Lock,並啟動子協程去 Unlock,主協程會阻塞在第二次 Lock 這兒等待子協程完成任務,但由於子協程足足睡眠了一秒,導致主協程等待這個鎖釋放足足等了一秒鐘。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 常量指針與指針常量 #include<iostream> using namespace std; int main() { int a = 10; int b = 20; // 常量指針與指針常量 // 1.常量指針 const修飾指針 指針的指向是可以修改的(指針變數中存的地址值可以修改) 指針 ...
  • 學習Spring源碼的建議 閱讀Spring官方文檔,瞭解Spring框架的基本概念和使用方法。 下載Spring源碼,可以從官網或者GitHub上獲取。 閱讀Spring源碼的入口類,瞭解Spring框架的啟動過程和核心組件的載入順序。 閱讀Spring源碼中的註釋和文檔,瞭解每個類和方法的作用和 ...
  • 平面上有n個點(n<=100),每個點的坐標均在-10000~10000之間,其中的一些點之間有連線。 若有連線,則表示可從一個點到達另一個點,即兩點間有通路,同路的距離為兩點間的直線距離。現在的任務是找出從一點到另一點之間的最短路徑。 小提示: 兩點的距離:如果點$A$坐標為$(x_A,y_A)$ ...
  • 本文首發於公眾號:Hunter後端 原文鏈接:Django筆記二十九之中間件介紹 這一節介紹一下 Django 的中間件。 關於中間件,官方文檔的解釋為:中間件是一個嵌入 Django 系統的 request 和 response 的鉤子框架,是一個能夠全局改變 Django 輸入/輸出的系統。 我 ...
  • 面向對象特性 封裝 在程式設計中,封裝(Encapsulation)是對具體對象的一種抽象,即將某些部分隱藏起來,在程式外部看不到,其含義是其他程式無法調用。要瞭解封裝,離不開“私有化”,就是將類或者是函數中的某些屬性限制在某個區域之內,外部無法調用。 封裝的作用: 1、保護隱私(把不想別人知道的東 ...
  • 原來的水文標題是“用 VS Code 搞 Qt6”,想想還是直接改為“Qt6”,反正這個用不用 VS Code 也能搞。雖然我知道大伙伴們都很討厭 CMake,但畢竟這廝幾乎成了 C++ 的玩家規範了。Qt 也算識大體,支持用 CMake 來構建程式。所以,只要你用的是能寫 C++ 的工具,理論上都 ...
  • 事實是這樣的,我有個介面,這個介面不能被篡改,於是想到了比較簡單的md5對url地址參數進行加密,把這個密碼當成是sign,然後服務端收到請求後,使用相同演算法也生成sign,兩個sign相同就正常沒有被篡改過。 問題的出現 介面中的參數包括userId,extUserId,時間,其中extUserI ...
  • 本篇文章主要介紹了,如何通過Nginx配置跨域,並覆蓋後端服務跨域配置。 先看下後端代碼跨域配置: 主要的目標是:不修改後端跨域配置代碼,來實現Nginx跨域指定功能變數名稱。 @Bean public CorsFilter corsFilter() { final UrlBasedCorsConfigur ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...