在Go中如何轉(zhuǎn)儲(chǔ)一個(gè)方法的GOSSAFUNC圖
Go 編譯器的 SSA 后端包含一種工具,可以生成編譯階段的 HTML 調(diào)試輸出。這篇文章介紹了如何為函數(shù)和方法打印 SSA 輸出。
讓我們從一個(gè)包含函數(shù)、值方法和指針方法的示例程序開始:
package main
import (
"fmt"
)
type Numbers struct {
vals []int
}
func (n *Numbers) Add(v int) {
n.vals = append(n.vals, v)
}
func (n Numbers) Average() float64 {
sum := 0.0
for _, num := range n.vals {
sum += float64(num)
}
return sum / float64(len(n.vals))
}
func main() {
var numbers Numbers
numbers.Add(200)
numbers.Add(43)
numbers.Add(-6)
fmt.Println(numbers.Average())
}
通過 GOSSAFUNC
環(huán)境變量控制 SSA 調(diào)試輸出。此變量含有要轉(zhuǎn)儲(chǔ)的函數(shù)的名稱。這不是函數(shù)的完全限定名。對(duì)于上面的 func main
,函數(shù)名稱為 main
而不是 main.main
。
% env GOSSAFUNC=main go build
runtime
dumped SSA to ../../go/src/runtime/ssa.html
t
dumped SSA to ./ssa.html
在這個(gè)例子中,GOSSAFUNC=main
同時(shí)匹配了 main.main
和一個(gè)名為 runtime.main
的函數(shù)。[1]這有點(diǎn)不走運(yùn),但是實(shí)際上可能沒什么大不了的,因?yàn)槿绻阋獙?duì)代碼進(jìn)行性能調(diào)整,它就不會(huì)出現(xiàn)在 func main
中的巨大的意大利面塊中。
你的代碼更有可能在方法中,你可能已經(jīng)看到這篇文章,并尋找能夠轉(zhuǎn)儲(chǔ)方法的 SSA 輸出。
要為指針方法 func (n *Numbers) Add
打印 SSA 調(diào)試,等效函數(shù)名為 (*Numbers).Add
:[2]
% env "GOSSAFUNC=(*Numbers).Add" go build
t
dumped SSA to ./ssa.html
要為值方法 func (n Numbers) Average
打印 SSA 調(diào)試,等效函數(shù)名為 (*Numbers).Average
,即使這是一個(gè)值方法:
% env "GOSSAFUNC=(*Numbers).Average" go build
t
dumped SSA to ./ssa.html
-
如果你沒有從源碼構(gòu)建 Go,那么
runtime
軟件包的路徑可能是只讀的,并且可能會(huì)收到錯(cuò)誤消息。請(qǐng)不要使用sudo
來(lái)解決此問題。 -
請(qǐng)注意 shell 引用 。