有人可以幫助我了解以下關于 golang 循環范圍和臨時函數的驚喜的“為什么”方面嗎?以下是一些更復雜的代碼的摘錄:package mainimport ( "fmt" )type Caller struct { call func()}func printer(val int) { fmt.Printf("the value is %v\n", val)}func main () { values := []int{1,2,3} var callers []Caller for _,val := range values { call := func() { printer(val) } callers = append(callers, Caller{call}) } for _, caller := range callers { caller.call() }}這產生了(對我來說)令人驚訝的結果:the value is 3the value is 3the value is 3如果我通過將范圍值循環體更改為此代碼來更改該代碼: theVal := val call := func() { printer(theVal) } callers = append(callers, Caller{call})然后我們得到了最初希望的結果:the value is 1the value is 2the value is 3從根本上說,一個有效,另一個無效——我很清楚這一點,可以試著記住這個成語。我希望有更多的理解,也許是 golang 的一點生活課。范圍規則和延遲執行是什么意思,這意味著循環變量保持最終的有效值并提交到循環期間構建的每個臨時函數?為什么值“val”沒有放入動態構建的函數“調用”中?我懷疑我對一些基本的東西感到困惑。即使看到一個工作版本,我也不確定我將來能否避免這樣的陷阱。如果您對“為什么”迭代值會有這樣的行為有建議,我很想聽聽(并提前感謝您)。
1 回答

慕桂英3389331
TA貢獻2036條經驗 獲得超8個贊
這也適用于你。這是來自FAQ的鏈接。
package main
import (
"fmt"
)
type Caller struct {
call func()
}
func printer(val int) {
fmt.Printf("the value is %v\n", val)
}
func main() {
values := []int{1, 2, 3}
var callers []Caller
for _, val := range values {
var call func()
func(v int) {
call = func() {
printer(v)
}
}(val)
callers = append(callers, Caller{call})
}
for _, caller := range callers {
caller.call()
}
}
另一種方法是將 val 的當前值綁定到每個閉包啟動時,您可以將其存儲在一個新變量中,然后使用它(您為解決它所做的方法)。
- 1 回答
- 0 關注
- 118 瀏覽
添加回答
舉報
0/150
提交
取消