這篇文章主要講解了“go語(yǔ)言閉包的知識(shí)點(diǎn)整理”,文中的講解內(nèi)容簡(jiǎn)單清晰,易于學(xué)習(xí)與理解,下面請(qǐng)大家跟著小編的思路慢慢深入,一起來(lái)研究和學(xué)習(xí)“go語(yǔ)言閉包的知識(shí)點(diǎn)整理”吧!
創(chuàng)新互聯(lián)公司于2013年開始,先為新民等服務(wù)建站,新民等地企業(yè),進(jìn)行企業(yè)商務(wù)咨詢服務(wù)。為新民企業(yè)網(wǎng)站制作PC+手機(jī)+微官網(wǎng)三網(wǎng)同步一站式服務(wù)解決您的所有建站問(wèn)題。
go閉包
在 Golang中,閉包是一個(gè)可以引用其作用域之外變量的函數(shù)。
換句話說(shuō),閉包是一個(gè)內(nèi)部函數(shù),它可以訪問(wèn)創(chuàng)建它的范圍內(nèi)的變量。即使外部函數(shù)完成執(zhí)行并且作用域被破壞,依然可以訪問(wèn)。
在深入研究閉包之前,需要了解什么是匿名函數(shù)。
匿名函數(shù)
顧名思義,匿名函數(shù)就是沒有名字的函數(shù)。
舉個(gè)例子,我們創(chuàng)建一個(gè)一般函數(shù)和匿名函數(shù)
package mainimport ( "fmt")func sayhi1() { // 一般函數(shù) fmt.Println("hello golang, I am Regular function")}func main() { sayhi1() sayhi2 := func() { //匿名函數(shù) fmt.Println("hello golang, I am Anonymous function") } sayhi2()}
結(jié)果輸出:
hello golang, I am Regular functionhello golang, I am Anonymous function
通過(guò)創(chuàng)建一個(gè)返回函數(shù)的函數(shù)來(lái)使用匿名函數(shù)。
package mainimport ( "fmt")func sayhello(s string) func() { return func() { fmt.Println("hello", s) }}func main() { sayhello := sayhello("golang") sayhello()}
結(jié)果輸出:
hello golang
常規(guī)函數(shù)和匿名函數(shù)唯一區(qū)別是匿名函數(shù)不是在包級(jí)別聲明的,它們被動(dòng)態(tài)地聲明,通常要么被使用,要么被遺忘,要么被分配給一個(gè)變量供以后使用。
閉包的本質(zhì)
閉包是包含自由變量的代碼塊,這些變量不在這個(gè)代碼塊內(nèi)或者任何全局上下文中定義,而是在定義代碼塊的環(huán)境中定義。由于自由變量包含在代碼塊中,所以只要閉包還被使用,那么這些自由變量以及它們引用的對(duì)象就不會(huì)被釋放,要執(zhí)行的代碼為自由變量提供綁定的計(jì)算環(huán)境。
閉包的價(jià)值在于可以作為函數(shù)對(duì)象或者匿名函數(shù),對(duì)于類型系統(tǒng)而言,這意味著不僅要表示數(shù)據(jù)還要表示代碼。支持閉包的多數(shù)語(yǔ)言都將函數(shù)作為第一級(jí)對(duì)象,就是說(shuō)這些函數(shù)可以存儲(chǔ)到變量中作為參數(shù)傳遞給其他函數(shù),最重要的是能夠被函數(shù)動(dòng)態(tài)創(chuàng)建和返回。
Golang中的閉包同樣也會(huì)引用到函數(shù)外的變量,閉包的實(shí)現(xiàn)確保只要閉包還被使用,那么被閉包引用的變量會(huì)一直存在。從形式上看,匿名函數(shù)都是閉包。
我們來(lái)看個(gè)閉包例子:
package mainimport ( "fmt")func caller() func() int { callerd := 1 sum := 0 return func() int { sum += callerd return sum }}func main() { next := caller() fmt.Println(next()) fmt.Println(next()) fmt.Println(next())}
結(jié)果輸出:
1 2 3
該例子中called 和sum 是自由變量,caller函數(shù)返回的匿名函數(shù)為自由變量提供了計(jì)算環(huán)境,匿名函數(shù)和自由變量組成的代碼塊其實(shí)就是閉包。在閉包函數(shù)中,只有匿名函數(shù)才能訪問(wèn)自由變量called和sum,而無(wú)法通過(guò)其他途徑訪問(wèn),因此閉包自由變量的安全性。
按照命令式語(yǔ)言的規(guī)則,caller函數(shù)只是返回了匿名函數(shù)的地址,但在執(zhí)行匿名函數(shù)時(shí)將會(huì)由于在其作用域內(nèi)找不到sum和called變量而出錯(cuò)。而在函數(shù)式語(yǔ)言中,當(dāng)內(nèi)嵌函數(shù)體內(nèi)引用到體外的變量時(shí),將會(huì)把定義時(shí)涉及到的引用環(huán)境和函數(shù)體打包成一個(gè)整體(閉包)返回。閉包的使用和正常的函數(shù)調(diào)用沒有區(qū)別。
現(xiàn)在我們給出引用環(huán)境的定義:在程序執(zhí)行中的某個(gè)點(diǎn)所有處于活躍狀態(tài)的約束所組成的集合,其中的約束指的是一個(gè)變量的名字和其所代表的對(duì)象之間的聯(lián)系。
所以我們說(shuō):閉包=函數(shù)+引用環(huán)境
其實(shí)我們可以將閉包函數(shù)看成一個(gè)類(C++),一個(gè)閉包函數(shù)調(diào)用就是實(shí)例化一個(gè)對(duì)象,閉包的自由變量就是類的成員變量,閉包函數(shù)的參數(shù)就是類的函數(shù)對(duì)象的參數(shù)。在該例子中,next可以看作是實(shí)例化的一個(gè)對(duì)象,next()可以看做是對(duì)象函數(shù)調(diào)用的返回值。
這讓我們想起了一句名言:對(duì)象是附有行為的數(shù)據(jù),而閉包是附有數(shù)據(jù)的行為
閉包使用的一些例子
利用閉包實(shí)現(xiàn)數(shù)據(jù)隔離
假設(shè)你想創(chuàng)建一個(gè)函數(shù),該函數(shù)可以訪問(wèn)即使在函數(shù)退出后仍然存在的數(shù)據(jù)。舉個(gè)例子,如果你想統(tǒng)計(jì)函數(shù)被調(diào)用的次數(shù),但不希望其他任何人訪問(wèn)該數(shù)據(jù)(這樣他們就不會(huì)意外更改它),你就可以用閉包來(lái)實(shí)現(xiàn)它:
package mainimport ( "fmt")func caller() func() int { callerd := 0 return func() int { callerd++ return callerd }}func main() { next := caller() fmt.Println(next()) fmt.Println(next()) fmt.Println(next())}
結(jié)果輸出:
1 2 3
利用閉包包裝函數(shù)和創(chuàng)建中間件
Go 中的函數(shù)是一等公民。這意味著您不僅可以動(dòng)態(tài)創(chuàng)建匿名函數(shù),還可以將函數(shù)作為參數(shù)傳遞給函數(shù)。例如,在創(chuàng)建 Web 服務(wù)器時(shí),通常會(huì)提供一個(gè)功能來(lái)處理Web 請(qǐng)求到特定的路由。
package mainimport ( "fmt" "net/http")func main() { http.HandleFunc("/hello", hello) http.ListenAndServe(":3000", nil)}func hello(w http.ResponseWriter, r *http.Request) { fmt.Fprintln(w, "Hello!
")}
在上面例子中,函數(shù) hello() 被傳遞給 http.HandleFunc() 函數(shù),并在該路由匹配時(shí)調(diào)用。
雖然這段代碼不需要閉包,但如果我們想用更多邏輯包裝我們的處理程序,閉包是非常有用的。一個(gè)完美的例子是我們可以通過(guò)創(chuàng)建中間件來(lái)在我們處理程序執(zhí)行之前或之后做一些其它的工作。
什么是中間件?
中間件基本上是可重用功能的一個(gè)奇特術(shù)語(yǔ),它可以在設(shè)計(jì)用于處理 Web 請(qǐng)求的代碼之前和之后運(yùn)行代碼。在 Go 中,這些通常是通過(guò)閉包來(lái)實(shí)現(xiàn)的,但在不同的編程語(yǔ)言中,可以通過(guò)其他方式來(lái)實(shí)現(xiàn)。
在編寫 Web 應(yīng)用程序時(shí)使用中間件很常見,而且它們不僅可用于計(jì)時(shí)器(您將在下面看到一個(gè)示例)。例如,中間件可用于編寫代碼驗(yàn)證用戶是否登錄過(guò)一次,然后將其應(yīng)用到你的所有會(huì)員專頁(yè)。
讓我們看看一個(gè)簡(jiǎn)單的計(jì)時(shí)器中間件在 Go 中是如何工作的。
package mainimport ( "fmt" "net/http" "time")func main() { http.HandleFunc("/hello", timed(hello)) http.ListenAndServe(":3000", nil)}func timed(f func(http.ResponseWriter, *http.Request)) func(http.ResponseWriter, *http.Request) { return func(w http.ResponseWriter, r *http.Request) { start := time.Now() f(w, r) end := time.Now() fmt.Println("The request took", end.Sub(start)) }}func hello(w http.ResponseWriter, r *http.Request) { fmt.Fprintln(w, "Hello!
")}
timed() 函數(shù)接受一個(gè)可以用作處理函數(shù)的函數(shù),并返回一個(gè)相同類型的函數(shù),但返回的函數(shù)與傳遞它的函數(shù)不同。返回的閉包記錄當(dāng)前時(shí)間,調(diào)用原始函數(shù),最后記錄結(jié)束時(shí)間并打印出請(qǐng)求的持續(xù)時(shí)間。同時(shí)對(duì)我們的處理程序函數(shù)內(nèi)部實(shí)際發(fā)生的事情是不可知的。
現(xiàn)在我們要做的就是將我們的處理程序包裝在 timed(handler) 中并將閉包傳遞給 http.HandleFunc() 函數(shù)調(diào)用。
利用閉包使用 sort 二分搜索
使用標(biāo)準(zhǔn)庫(kù)中的包也經(jīng)常需要閉包,例如 sort 包。這個(gè)包為我們提供了大量有用的功能和代碼,用于排序和搜索排序列表。例如,如果您想對(duì)一個(gè)整數(shù)切片進(jìn)行排序,然后在切片中搜索數(shù)字 7,您可以像這樣使用 sort 包。
package mainimport ( "fmt" "sort")func main() { numbers := []int{1, 11, -5, 7, 2, 0, 12} sort.Ints(numbers) fmt.Println("Sorted:", numbers) index := sort.SearchInts(numbers, 7) fmt.Println("7 is at index:", index)}
結(jié)果輸出:
Sorted: [-5 0 1 2 7 11 12]7 is at index: 4
如果要搜索的每個(gè)元素都是自定義類型的切片會(huì)發(fā)生什么?或者,如果您想找到第一個(gè)等于或大于 7 的數(shù)字的索引,而不僅僅是 7 的第一個(gè)索引?
為此,您可以使用 sort.Search() 函數(shù),并且您需要傳入一個(gè)閉包,該閉包可用于確定特定索引處的數(shù)字是否符合您的條件。
sort.Search() is a binary search
sort.Search 函數(shù)執(zhí)行二分搜索,因此它需要一個(gè)閉包,該閉包在滿足您的條件之前對(duì)任何索引返回 false,在滿足后返回 true。
讓我們使用上面描述的示例來(lái)看看它的實(shí)際效果;我們將搜索列表中第一個(gè)大于或等于 7 的數(shù)字的索引。
package mainimport ( "fmt" "sort")func main() { numbers := []int{1, 11, -5, 8, 2, 0, 12} sort.Ints(numbers) fmt.Println("Sorted:", numbers) index := sort.Search(len(numbers), func(i int) bool { return numbers[i] >= 7 }) fmt.Println("The first number >= 7 is at index:", index) fmt.Println("The first number >= 7 is:", numbers[index])}
結(jié)果輸出:
Sorted: [-5 0 1 2 8 11 12]The first number >= 7 is at index: 4 The first number >= 7 is: 8
在這個(gè)例子中,我們的閉包是作為第二個(gè)參數(shù)傳遞給 sort.Search() 的簡(jiǎn)單函數(shù)。
這個(gè)閉包訪問(wèn)數(shù)字切片,即使它從未被傳入,并為任何大于或等于 7 的數(shù)字返回 true。通過(guò)這樣做,它允許 sort.Search() 工作而無(wú)需了解什么您使用的基礎(chǔ)數(shù)據(jù)類型是什么,或者您試圖滿足什么條件。它只需要知道特定索引處的值是否符合您的標(biāo)準(zhǔn)。
用閉包+defer進(jìn)行處理異常
package mainimport ( "fmt")func handle() { defer func() { err := recover() if err != nil { fmt.Println("some except had happend:", err) } }() var a *int = nil *a = 100}func main() { handle()}
結(jié)果輸出:
some except had happend: runtime error: invalid memory address or nil pointer dereference
recover函數(shù)用于終止錯(cuò)誤處理流程。一般情況下,recover應(yīng)該在一個(gè)使用defer關(guān)鍵字的函數(shù)中執(zhí)行以有效截取錯(cuò)誤處理流程。如果沒有在發(fā)生異常的goroutine中明確調(diào)用恢復(fù)過(guò)程(調(diào)用recover函數(shù)),會(huì)導(dǎo)致該goroutine所屬的進(jìn)程打印異常信息后直接退出
對(duì)于第三方庫(kù)的調(diào)用,在不清楚是否有panic的情況下,最好在適配層統(tǒng)一加上recover過(guò)程,否則會(huì)導(dǎo)致當(dāng)前進(jìn)程的異常退出,而這并不是我們所期望的。
感謝各位的閱讀,以上就是“go語(yǔ)言閉包的知識(shí)點(diǎn)整理”的內(nèi)容了,經(jīng)過(guò)本文的學(xué)習(xí)后,相信大家對(duì)go語(yǔ)言閉包的知識(shí)點(diǎn)整理這一問(wèn)題有了更深刻的體會(huì),具體使用情況還需要大家實(shí)踐驗(yàn)證。這里是創(chuàng)新互聯(lián),小編將為大家推送更多相關(guān)知識(shí)點(diǎn)的文章,歡迎關(guān)注!