Golang?并發(fā)下的問題定位及解決方案
問題描述
在使用 gin-swagger 的過程中, 經(jīng)常會(huì)發(fā)生因?yàn)槿鄙?json 等 tag 而導(dǎo)致的異常。 由于 gin-swagger 是并發(fā)執(zhí)行的, 輸出的日志本身是錯(cuò)位的。 這就導(dǎo)致無法定義是哪一個(gè)結(jié)構(gòu)體缺少 tag 導(dǎo)致的。

一般而言, 這種時(shí)候只能一個(gè)個(gè)點(diǎn)開去檢查。
解決方案
思路 : 要是每行日志帶當(dāng)前 goroutine_id 的話, 是不是就可以準(zhǔn)確定位到報(bào)錯(cuò)的 goroutine 他打印的日志是哪些了呢!
說做就做
實(shí)現(xiàn)思路
- 查看當(dāng)前日志是怎么打印的
發(fā)現(xiàn) gin-swagger 日志直接調(diào)用的 golang 的標(biāo)準(zhǔn)庫 log
由于沒有對(duì)log初始化, 所以默認(rèn)使用的是 stdout 。
log.Printf("Picking operation from %s\n", aurora.Blue(funType.FullName()))- 如果想要給日志中添加
goroutine_id的話, 就需要在調(diào)用log.Printf的時(shí)候獲取當(dāng)前goroutine的 id , 所以首先要解決的是怎么獲取goroutine_id的問題。
調(diào)研后發(fā)現(xiàn)了集中常見的獲取 goroutine_id 的方法:
2.1 通過棧信息解析后獲取
func GetGID() uint64 {
b := make([]byte, 64)
b = b[:runtime.Stack(b, false)]
b = bytes.TrimPrefix(b, []byte("goroutine "))
b = b[:bytes.IndexByte(b, ' ')]
n, _ := strconv.ParseUint(string(b), 10, 64)
return n
}2.2 修改 Go 源碼獲取
# src/runtime/runtime2.go
func Goid() int64 {
_g_ := getg()
return _g_.goid
}2.3 通過 CGO 獲取
文件 id.c
#include "runtime.h"
int64 ·Id(void) {
return g->goid;
}文件 id.go
package id func Id() int64
另外還可以通過匯編獲取 goroutine_id
由于go的版本不同,goroutine的結(jié)構(gòu)也可能不同, 所以此處我直接調(diào)用一個(gè)開源實(shí)現(xiàn):
https://github.com/petermattis/goid
- 修改
gin-swaggermain.go源碼, 修改logWrite的實(shí)現(xiàn)
修改前
func main() {
cmd.Execute()
}修改后
type Out os.File
func (o Out) Write(b []byte) (int, error) {
prefix := fmt.Sprintf("gid-%d: ", goid.Get())
all := make([]byte, len(b)+len(prefix))
all = []byte(prefix)
all = append(all, b...)
f := os.File(o)
return f.Write(all)
}
func main() {
var out Out
out = Out(os.Stdout)
log.SetOutput(out)
cmd.Execute()這樣修改后,每次 gin-swagger 調(diào)用 log 打印日志都時(shí)候, 都會(huì)使用我定義的 Write 方法, Write 方法中每次打印前都會(huì)先查詢出當(dāng)前的 goroutine_id ,然后作為日志的前綴。
修改后的日志效果

從中可以看到每行日志開頭,都已經(jīng)加上了 goroutine_id。 只要使用 panic goroutine 的 id 做一次 grep , 即可篩選出需要的日志了,極大的方便了定位問題。

cat /tmp/gin-swagger.log | grep 7647

到此這篇關(guān)于Golang 并發(fā)下的問題定位的文章就介紹到這了,更多相關(guān)Golang并發(fā)問題內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
go?module化?import?調(diào)用本地模塊?tidy的方法
這篇文章主要介紹了go?module化?import?調(diào)用本地模塊?tidy的相關(guān)知識(shí),本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-09-09
go如何終止多個(gè)for select循環(huán)嵌套的方法
當(dāng)您想從嵌套循環(huán)中中斷,從select內(nèi)部終止循環(huán)時(shí),標(biāo)記的中斷非常有用,本文主要介紹了go如何終止多個(gè)for select循環(huán)嵌套的方法,具有一定的參考價(jià)值,感興趣的可以了解一下2024-01-01
go build 通過文件名后綴實(shí)現(xiàn)不同平臺(tái)的條件編譯操作
這篇文章主要介紹了go build 通過文件名后綴實(shí)現(xiàn)不同平臺(tái)的條件編譯操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-12-12
Go程序的init函數(shù)在什么時(shí)候執(zhí)行
在Go語言中,init?函數(shù)是一個(gè)特殊的函數(shù),它用于執(zhí)行程序的初始化任務(wù),本文主要介紹了Go程序的init函數(shù)在什么時(shí)候執(zhí)行,感興趣的可以了解一下2023-10-10
go將request?body綁定到不同的結(jié)構(gòu)體中教程
這篇文章主要為大家介紹了go將request?body綁定到不同的結(jié)構(gòu)體中教程,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-10-10

