在 Go 中,「defer」語句允許我們安排一個函數在周圍函數傳回之前執行。但是,在處理在封閉函數中修改的變數時,這可能會導致混亂。
考慮以下函數:
func printNumbers() { var x int defer fmt.Println(x) for i := 0; i < 5; i++ { x++ } }
根據Go規範中,當執行「defer」語句時,函數值和參數將被評估並儲存以以後執行。這意味著當函數最終被呼叫時,x 的值仍將為 0,因為它是在延遲時計算的。
為了解決這個問題,我們可以在「defer」語句內使用匿名函數:
defer func() { fmt.Println(x) }()
這裡,x不是匿名函數的參數,所以不會當執行「defer」語句時評估。相反,x 的值將在呼叫匿名函數時捕獲,確保列印最新的值。
使用指標:
var x int defer func() { fmt.Println(&x) }()
此方法使用指向x 的指標作為延遲函數的函數的函數參數。執行「defer」語句時,僅計算指針,而不計算 x 的值。當延遲函數被呼叫時,它將透過指標存取 x 的當前值。
使用自訂類型:
type MyInt int func (m *MyInt) String() string { return strconv.Itoa(int(*m)) } var x MyInt defer fmt.Println(&x)
此解決方案類似於指標方法,但使用實作String() 方法的自訂類型(MyInt)。透過實作 String(),我們可以控制 x 的值如何列印。
使用切片:
var x []int defer fmt.Println(x)
切片是Go 中的描述符類型,這意味著它的值是對底層數組的引用。當我們延遲切片時,僅評估引用,而不評估實際的陣列元素。因此,延遲後對切片元素所做的任何更改都將反映在列印輸出中。
包裝在結構中:
type Wrapper struct { Value int } var x Wrapper defer fmt.Println(&x)
這種方法類似於使用指針,但我們將值包裝在結構中以避免在結構體中取消引用指標延遲函數。
以上是如何在 Go 中正確使用'defer”和修改變數?的詳細內容。更多資訊請關注PHP中文網其他相關文章!