Golang中的閉包(Closures)詳解
在講解閉包之前先看一下 Golang 中的匿名函數。
匿名函數(Anonymous Functions)
匿名函數也可以稱為函數字面量、lambda 函數或者閉包。閉包的概念起源于 lambda 微積分中表達式的數學求值。從技術上講,匿名函數和閉包之間有細微的區(qū)別:匿名函數是沒有名稱的函數,而閉包則是函數的實例。在 Golang 中要實現閉包,是離不開匿名函數的。
先看一個普通函數的例子,例如:
func add(x, y int) { fmt.Println(x + y) }
調用方式如下:
add(1, 2) // 輸出 3
接下來看下如何使用匿名函數來實現相同的功能:
func(x, y int) { fmt.Println(x + y) }(1, 2)
這個匿名函數和上面的普通的函數的功能是一樣的,區(qū)別是
- 沒有名字
- 定義之后就直接調用
接下來,使用通過創(chuàng)建一個返回一個函數的函數的方式來使用一個匿名函數。函數一般都是返回整數、字符串、結構體等基本類型,但是在 Golang 中一個函數可以返回另一個函數。如下是 Golang 官方的一個例子:
func adder() func(int) int { sum := 0 return func(x int) int { sum += x return sum } }
這個函數的返回類型是 func(int) int 類型的函數,可以將這個函數的返回值賦值給一個變量,然后可以像調用一個函數的方式使用調用這個變量,例如:
pos := adder() pos(1)
閉包(Closures)
通過上文的講解我們已經知道了匿名函數的定義以及使用方式,也了解了一個函數可以返回另一個函數,接下來講解下閉包。
在 Golang 中,閉包是一個引用了作用域之外的變量的函數。閉包的存在時間可以超過創(chuàng)建它的作用域,因此它可以訪問該作用域中的變量,即使在該作用域被銷毀之后。上文中的 adder() 函數返回的就是一個典型的閉包。
Golang 中的匿名函數也被稱為閉包,匿名函數是一種特殊類型的函數,沒有名稱,閉包可以被認為是一種特殊類型的匿名函數。
Golang 中的閉包由兩部分組成:函數體和函數執(zhí)行時的上下文環(huán)境。函數體定義了閉包的邏輯,上下文環(huán)境則包含了函數外部的變量。當閉包被創(chuàng)建時,會將外部變量的引用保存在上下文環(huán)境中,并且在函數體內部可以隨時訪問這些外部變量??磦€將上文中的 adder() 函數稍作修改的例子:
package main import "fmt" func adder() func(int) int { sum := 0 return func(x int) int { fmt.Println("執(zhí)行前 sum =", sum) sum += x return sum } } func main() { pos := adder() for i := 0; i < 4; i++ { fmt.Println("執(zhí)行后 sum =", pos(1)) } }
運行結果如下:
執(zhí)行前 sum = 0
執(zhí)行后 sum = 1
執(zhí)行前 sum = 1
執(zhí)行后 sum = 2
執(zhí)行前 sum = 2
執(zhí)行后 sum = 3
執(zhí)行前 sum = 3
執(zhí)行后 sum = 4
可以看出,閉包函數引用的外部變量被保存在了上下文環(huán)境中(一直不被銷毀),每次執(zhí)行閉包,閉包內的變量又保存了上一次運行后的值。
小結
閉包是來源于函數式編程語言的一種特性,函數既可以返回一個函數、也可以接受一個函數作為參數(這種函數被稱為高階函數)。Golang也支持函數式編程,閉包在Golang 中有非常廣泛的使用,并且經常與Goroutine 和 channel 一起使用。
到此這篇關于Golang中的閉包(Closures)詳解的文章就介紹到這了,更多相關Golang閉包(Closures)內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
基于golang channel實現的輕量級異步任務分發(fā)器示例代碼
這篇文章主要給大家介紹了關于基于golang channel實現的輕量級異步任務分發(fā)器的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2018-07-07