php小編魚仔封閉所容納的環境範圍是指封閉環境中所包含的元素、因素和條件。封閉環境是指一個相對封閉、受限的空間或範圍,可以是實體空間,如實驗室、工廠車間, 也可以是虛擬空間,如電腦網路。在封閉環境中,各種因素如溫度、濕度、氣體組成等都可以被控制和調節,以達到特定的目的。封閉環境常見於科學研究實驗、生產製造等領域,對於確保實驗精確度、產品品質等方面扮演重要角色。
我模仿網路上關於閉包的教程,寫了下面的程式碼。
func foo1() func() { xvalue := 1 x := &xvalue defer func() { xvalue = 2 }() return func() { *x = *x + 1 fmt.printf("foo1 val = %d\n", *x) } } func main() { f1 := foo1() f1() f1() f1() }
我很困惑,在執行f1 := foo1()
後,變數xvalue
似乎應該被回收,因此使用*x
應該是錯誤的,但上面的程式碼沒有錯誤並且執行罰款,給出輸出
foo1 val = 3 foo1 val = 4 foo1 val = 5
所以我想知道閉包除了保存指標本身之外還保存了指標的值還是go語言的垃圾回收機制導致xvalue沒有被刪除?
在 Go 中,閉包取得對其關閉的任何變數(的位址)的參考。引用語言參考:
函數文字是閉包:它們可以引用周圍函數中定義的變數。然後,這些變數在周圍的函數和函數文字之間共享,並且只要可訪問,它們就會一直存在。
因此,在您的範例中:
f1 := foo1()
xValue
變數存在(編譯器可能會在堆上分配它)。它將以其類型的零值 0 開始。 x
存在並為其指派 xValue
的位址。 defer
-red 閉包運行並將值 2 指派給 xValue
。 x
的閉包。 後一點可能有點棘手:由於返回的閉包引用了變數 x
,編譯器保證即使在 foo
返回後該變數也存在。由於 x
包含 xValue
的位址(因此是對它的即時引用),因此該位址仍然存在,並且不能被垃圾收集。
使用相同的轉義分析方法,編譯器保證 xValue
在其宣告的函數傳回後仍然存在。
您執行傳回的閉包,該閉包透過指向它的指標來修改 xValue
# – 這裡沒有發生任何魔法。另外兩個調用執行相同的操作。
總而言之,也許您被C 知識絆倒了,一旦從該函數返回控制權,函數中聲明的任何變數都將不再存在,因此該函數外部存在的對該變數的任何引用都將變為無效的。在Go 中,情況並非如此:在這方面,語言被明確定義為安全:編譯器確保任何變數都有適當的分配,以便在返回(或以其他方式傳達)對它的引用時在創建它的函數呼叫中存活下來。從該函數呼叫到外部世界。
以上是封閉所容納的環境範圍是什麼?的詳細內容。更多資訊請關注PHP中文網其他相關文章!