go的defer和閉包示例說明(非內(nèi)部實(shí)現(xiàn))
golang的閉包函數(shù)
用幾個例子說明golang的閉包函數(shù),結(jié)合defer使用,配合對應(yīng)代碼及文末總結(jié)。
| 函數(shù) | 說明 | 輸出 |
|---|---|---|
| e1 | defer調(diào)用,相當(dāng)于是拿到了當(dāng)前err變量的快照,即注冊defer函數(shù)的時候,將當(dāng)下err的值塞入到defer中 | start err1 |
| e2 | defer 調(diào)用,但是一個閉包函數(shù),且閉包函數(shù)有傳參,閉包捕獲當(dāng)前err的值仍然是 start err2(閉包捕獲的是變量值的拷貝),且閉包內(nèi)的值變量改變不會影響外部err的值(詳見見e5) | start err2 |
| e3 | defer 調(diào)用,閉包內(nèi)的變量和匿名函數(shù)外的變量是公用的,沒有傳遞形參,沒有傳遞形參,與上下文共享 | defer3 error |
| e4 | defer 調(diào)用,在函數(shù) e4 中,當(dāng)你將 err 作為參數(shù)傳遞給閉包函數(shù)時,實(shí)際上是創(chuàng)建了一個閉包函數(shù)的副本,這個副本在閉包內(nèi)部獨(dú)立于外部作用域。這種行為是因?yàn)殚]包在捕獲外部變量時,會將外部變量的當(dāng)前值復(fù)制到閉包內(nèi)部,形成一個閉包環(huán)境,現(xiàn)在理解了閉包的概念了吧。具體來說,在 defer 語句執(zhí)行的時候,閉包函數(shù)會將 err 的當(dāng)前值(即 "start err4")復(fù)制到閉包內(nèi)部的參數(shù)中。之后,無論外部作用域的 err 是否發(fā)生改變,閉包內(nèi)部的參數(shù)值都會保持不變,因?yàn)殚]包已經(jīng)捕獲了一個快照 | start err4 |
| e5 | 傳值的情況下,閉包內(nèi)的值變量改變不會影響外部err的值,(互相獨(dú)立) | now err is start err5 start err5CHANGE ME |
| e6 | 閉包沒有傳值,拿到的err是最后賦值的, | now err is start err6 defer6 error CHANGE ME |
package main
import (
"errors"
"fmt"
)
func e1(){
err := errors.New("start err1")
defer fmt.Println(err)
err = errors.New("defer1 error")
return
}
func e2(){
err := errors.New("start err2")
defer func(e error) {
fmt.Println(e)
}(err)
err = errors.New("defer2 error")
return
}
func e3(){
err := errors.New("start err3")
//閉包內(nèi)的變量和匿名函數(shù)外的變量是公用的,沒有傳遞形參,沒有傳遞形參,與上下文共享
defer func() {
fmt.Println(err)
}()
err = errors.New("defer3 error")
return
}
func e4(){
var err error
err = errors.New("start err4")
//閉包內(nèi)的變量和匿名函數(shù)外的變量是公用的,但是如果傳了形參,那就和上文的共用了
//在函數(shù) e4 中,當(dāng)你將 err 作為參數(shù)傳遞給閉包函數(shù)時,實(shí)際上是創(chuàng)建了一個閉包函數(shù)的副本,這個副本在閉包內(nèi)部獨(dú)立于外部作用域。這種行為是因?yàn)殚]包在捕獲外部變量時,會將外部變量的當(dāng)前值復(fù)制到閉包內(nèi)部,形成一個閉包環(huán)境
//具體來說,在 defer 語句執(zhí)行的時候,閉包函數(shù)會將 err 的當(dāng)前值(即 "start err4")復(fù)制到閉包內(nèi)部的參數(shù)中。之后,無論外部作用域的 err 是否發(fā)生改變,閉包內(nèi)部的參數(shù)值都會保持不變,因?yàn)殚]包已經(jīng)捕獲了一個快照。
defer func(err error) {
fmt.Println(err)
}(err)
err = errors.New("defer4 error")
return
}
func e5(){
err := errors.New("start err4")
defer func(err error ) {
err=errors.New(err.Error()+"CHANGE ME")
fmt.Println(err)
}(err)
fmt.Println("now err is ",err)
err = errors.New("defer5 error")
return
}
func e6() {
err := errors.New("start err6")
defer func() {
err = errors.New(err.Error() + " CHANGE ME")
fmt.Println(err)
}()
fmt.Println("now err is ", err)
err = errors.New("defer6 error")
return
}
func main(){
e1()
e2()
e3()
e4()
e5()
e6()
}變量作用域和閉包
Go 語言中的變量作用域由代碼塊決定。變量在其定義的代碼塊內(nèi)可見。
閉包是一個函數(shù)值,它可以捕獲其定義時周圍的作用域內(nèi)的變量。
閉包可以在定義之外被調(diào)用,仍然訪問并修改捕獲的變量。
閉包和變量捕獲
閉包函數(shù)可以捕獲外部作用域的變量。在閉包內(nèi)部,它們可以訪問外部變量的值。
閉包捕獲的變量是其副本,即閉包內(nèi)部使用的是變量值的拷貝。
修改閉包內(nèi)部捕獲的變量不會影響外部作用域中的變量,除非你在閉包內(nèi)直接修改外部作用域的變量。
閉包參數(shù)傳遞
在閉包內(nèi)部接收外部作用域的變量作為參數(shù),可以使閉包操作外部作用域的變量。
使用閉包參數(shù)傳遞可以有效隔離閉包內(nèi)外的變量,從而保持可預(yù)測性。
在 defer 中的閉包
當(dāng)在 defer 語句中使用閉包時,閉包內(nèi)部的變量會被“捕獲”并在 defer 執(zhí)行時使用。
在閉包內(nèi)部修改閉包捕獲的變量不會影響外部作用域中的變量,除非你直接修改外部作用域的變量。
總結(jié)
閉包是一種強(qiáng)大的概念,可以使函數(shù)擁有狀態(tài)并延遲執(zhí)行。
了解閉包如何操作變量作用域,以及它們?nèi)绾尾东@和修改變量,是編寫高效、清晰的 Go 代碼的關(guān)鍵。
當(dāng)在閉包中操作變量時,要注意變量作用域、捕獲的變量副本和對外部作用域的影響。
以上就是go的defer和閉包示例說明(非內(nèi)部實(shí)現(xiàn))的詳細(xì)內(nèi)容,更多關(guān)于go defer閉包的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
jenkins配置golang?代碼工程自動發(fā)布的實(shí)現(xiàn)方法
這篇文章主要介紹了jenkins配置golang?代碼工程自動發(fā)布,jks是個很好的工具,使用方法也很多,我只用了它簡單的功能,對jenkins配置golang相關(guān)知識感興趣的朋友一起看看吧2022-07-07
Go到底能不能實(shí)現(xiàn)安全的雙檢鎖(推薦)
這篇文章主要介紹了Go到底能不能實(shí)現(xiàn)安全的雙檢鎖,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-05-05
Go/C語言LeetCode題解997找到小鎮(zhèn)法官
這篇文章主要為大家介紹了Go語言LeetCode題解997找到小鎮(zhèn)的法官示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-12-12

