我嘗試解決我的泄漏代碼。但是向通道添加緩沖區并沒有達到目的。我的代碼package mainimport (? ? "fmt"? ? "runtime"? ? "time")func main() {? ? fmt.Println(runtime.NumGoroutine())? ? leaking()? ? time.Sleep(5)? ? fmt.Println(runtime.NumGoroutine())}func leaking() {? ? errChang := make(chan int, 1)? ? go func() {? ? ? ? xx :=? return666()? ? ? ? errChang <- xx? ? }()? ? fmt.Println("hola")? ? return? ? fmt.Println(<-errChang)}func return666() int {? ? time.Sleep(time.Second * 1)? ? return 6}我最初的代碼沒有使用緩沖區,導致 go-routine 中的泄漏函數,.. 泄漏。在這篇文章之后,我預計通過向通道添加緩沖區,可以避免泄漏。
1 回答

撒科打諢
TA貢獻1934條經驗 獲得超2個贊
這里,在 Go Playground 中,是經過一些細微修改的原始代碼:
延遲減少,除了
time.Sleep(5)
變為time.Sleep(time.Second)
;a
return
被刪除,因為它變得不必要;a
fmt.Println
被注釋掉,因為 thereturn
和未注釋的fmt.Println
都會go vet
抱怨無法訪問fmt.Println
;存儲的通道
errChang
更改為無緩沖。
運行時,其輸出為:
1 hola 2
(在 之前有一個小的延遲2
),表明您在 function 中啟動的匿名 goroutine 確實leaking
仍在運行。
如果我們取消注釋掉的注釋fmt.Println
,則輸出為:
1 hola 6 1
(在final之前有同樣的輕微延遲1
)因為我們現在等待(然后打?。┰?code>return666channel中計算并發送的值errChang
。
如果我們保留注釋掉的注釋fmt.Println
并使通道緩沖,則輸出將變為:
1 hola 1
因為匿名 goroutine 現在能夠將其值 (6) 推送到通道中。
通道本身以及存儲在其中的單個值將被垃圾收集,因為此時沒有對該通道的剩余引用。但請注意,僅僅使通道緩沖并不總是足夠的。如果我們沿著通道發送兩個值,程序將返回打印:
1 hola 2
因為匿名 goroutine 成功放入6
通道,但隨后也阻止了嘗試放入42
。
- 1 回答
- 0 關注
- 132 瀏覽
添加回答
舉報
0/150
提交
取消