在Go語言中,函數是一等(first-class)公民,函數類型也是一等的數據類型。
函數不但可以用于封裝代碼、分割功能、解耦邏輯,還可以化身為普通的值,在其他函數間傳遞、賦予變量、做類型判斷和轉換等等。函數值可以由此成為能夠被隨意傳播的獨立邏輯組件(或者說功能模塊)。
開頭說的,函數是一等公民,函數類型是一等數據類型:
package main
import "fmt"
type calcFunc func(int, int) int
func add(x, y int) int {
return x + y
}
func sub(x, y int) int {
return x - y
}
func main() {
var f1, f2 calcFunc
f1, f2 = add, sub
fmt.Println(f1(1, 2))
fmt.Println(f2(100, 50))
}
這里,先聲明了一個函數類型。在下面聲明的兩個函數的前面與caleFunc是一致的。因此都是caleFunc的一個實現。在main函數中,分別把兩個函數賦值給caleFunc類型的變量f1和f2,然后可以調用它們。
這里書寫函數簽名的方式與函數聲明的是一致的。只是緊挨在參數列表左邊的不是函數名稱,而是關鍵字func。這里函數名稱和func互換了一下位置而已。
函數簽名,就是函數的參數列表和結果列表的統稱,它定義了可用來鑒別不同函數的那些特征,同時也定義了我們與函數交互的方式。
“函數是一等的公民”是函數式編程(functional programming)的重要特征。Go語言在語言層面支持了函數式編程。
高階函數可能具有如下兩個特點:
函數只要滿足了上面的任意一個特點,就可以說這個函數是一個高階函數。高階函數也是函數式編程中的重要概念和特征。
在上面的例子的基礎上,寫一個高階函數,然后再main主函數里調用執行:
package main
import (
"errors"
"fmt"
)
type calcFunc func(int, int) int
func add(x, y int) int {
return x + y
}
func sub(x, y int) int {
return x - y
}
func calculate(x, y int, f calcFunc) (int, error) {
if f == nil {
return 0, errors.New("invalid calcFunc")
}
return f(x, y), nil
}
func main() {
res, err := calculate(1, 2, add)
if err != nil {
fmt.Println("ERROR:", err)
}
fmt.Println("Result:", res)
}
上面的calculate就是一個高階函數,“接收其他的函數作為參數傳入”的這種高階函數。下面的示例是另外一種高階函數“把其他的函數作為結果返回”:
package main
import (
"errors"
"fmt"
)
type calcFunc func(int, int) int
func add(x, y int) int {
return x + y
}
func sub(x, y int) int {
return x - y
}
type resFunc func(int, int) (int, error)
func genCalculator(f calcFunc) resFunc {
return func(x, y int) (int, error) {
if f == nil {
return 0, errors.New("invalid calcFunc")
}
return f(x, y), nil
}
}
func main() {
f := genCalculator(sub)
res, _ := f(10, 6)
fmt.Println("Result:", res)
}
自由變量,在一個函數中存在對外來標識符的引用。所謂的外來標識符,是既不代表當前函數的任何參數或結果,也不是函數內部聲明的,它是直接從外邊拿過來的。
上面的例子中的genCalculator函數內部,實際上就實現了一個閉包。而genCalculator函數也是一個高階函數:
func genCalculator(f calcFunc) resFunc {
return func(x, y int) (int, error) {
if f == nil {
return 0, errors.New("invalid calcFunc")
}
return f(x, y), nil
}
}
genCalculator函數只做了一件事,那就是定義一個匿名的函數并把它作為結果值返回。而這個匿名的函數就是一個閉包函數。它里面使用的變量f既不代表它的任何參數或結果也不是它自己聲明的,而是定義它的genCalculator函數的參數,所以是一個自由變量。而自由變量具體是什么,并不是在定義閉包的時候確定的,而是在genCalculator函數被調用的時候確定的。
函數是Go語言支持函數式編程的主要體現。我們可以通過“把函數傳給函數”以及“讓函數返回函數”來編寫高階函數,也可以用高階函數來實現閉包,并以此做到部分程序邏輯的動態生成。
最后,還有一個閉包的典型例子:
package main
import "fmt"
func increment () func() {
var x int
return func () {
x ++
fmt.Println(x)
}
}
func main() {
f := increment()
f() // 打印1
f() // 打印2
f() // 打印3
}
主函數里,每次調用執行的結果都會變化。變量x屬于閉包的一部分,但是是在閉包的函數外的,每次調用閉包后的x的狀態都會保留下來。
另外有需要云服務器可以了解下創新互聯scvps.cn,海內外云服務器15元起步,三天無理由+7*72小時售后在線,公司持有idc許可證,提供“云服務器、裸金屬服務器、高防服務器、香港服務器、美國服務器、虛擬主機、免備案服務器”等云主機租用服務以及企業上云的綜合解決方案,具有“安全穩定、簡單易用、服務可用性高、性價比高”等特點與優勢,專為企業上云打造定制,能夠滿足用戶豐富、多元化的應用場景需求。
文章標題:Go36-12-函數-創新互聯
文章出自:http://vcdvsql.cn/article30/hocso.html
成都網站建設公司_創新互聯,為您提供用戶體驗、網站設計、網站收錄、關鍵詞優化、ChatGPT、電子商務
聲明:本網站發布的內容(圖片、視頻和文字)以用戶投稿、用戶轉載內容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網站立場,如需處理請聯系客服。電話:028-86922220;郵箱:631063699@qq.com。內容未經允許不得轉載,或轉載時需注明來源: 創新互聯