Go 語言的函數(shù)與指針
函數(shù)
函數(shù)就是一段基本的代碼塊,一般用來對(duì)需要重復(fù)執(zhí)行的代碼進(jìn)行復(fù)用。在 go 中,函數(shù)是『一等公民』,這與 js 類似,也就是可以將函數(shù)當(dāng)做一個(gè)變量進(jìn)行傳遞。
函數(shù)聲明
由于是強(qiáng)類型語言,與 js 不同,在函數(shù)聲明的過程中,需要指定參數(shù)與返回值的類型。
- func max (n1, n2 int) int {
- var result int
- if n1 >= n2 {
- result = n1
- }
- if n1 < n2 {
- result = n2
- }
- return result
- }
在聲明函數(shù)參數(shù)和類型的時(shí)候,與聲明變量類似,可以一次性指定多個(gè)參數(shù)的類型,也可以分別指定多個(gè)參數(shù)為不同類型。
- func max (n1 int, n2 int) int {
- ……
- }
如果函數(shù)返回值有多個(gè),在指定返回類型的時(shí)候,需要指定每個(gè)返回值的類型。
- func max (n1 int, n2 int) (error, int) {
- ……
- return errors.New(""), result
- }
上面的代碼,表示返回的時(shí)候需要返回兩個(gè)值,第一個(gè)值為 error 對(duì)象,用來表示執(zhí)行期間是否出現(xiàn)異常。這種方式也是 Node.js 中常見的 error-first callback 的寫法。
特殊函數(shù)
在 go 中,有兩個(gè)特殊函數(shù):main、init,這兩個(gè)函數(shù)聲明之后,一般不需要主動(dòng)調(diào)用,會(huì)有自動(dòng)執(zhí)行的機(jī)制。
func main()
main 函數(shù)是 go 語言中默認(rèn)的入口函數(shù),只能應(yīng)用于 package main 中,如果在其他的 package 中不會(huì)執(zhí)行。main 函數(shù)有如下幾點(diǎn)需要注意:
- 不能定義參數(shù);
- 不能定義返回值;
- 必須在 package main 中聲明;
func init()
init 函數(shù)所有的包啟動(dòng)的時(shí)候都會(huì)執(zhí)行,執(zhí)行時(shí)機(jī)比 main 函數(shù)早,與 main 函數(shù)一樣,不能定義參數(shù)和返回值。
- package main
- import "fmt"
- func init() {
- fmt.Println("執(zhí)行 init 函數(shù)\n")
- }
- func main() {
- fmt.Println("執(zhí)行 main 函數(shù)\n")
- }
函數(shù)調(diào)用
函數(shù)的調(diào)用比較簡單,和其他編程語言類似,只需要將函數(shù)需要接受的參數(shù)傳入其中,在執(zhí)行結(jié)束后,就能得到對(duì)應(yīng)的返回值。
- // 定義 max 函數(shù)
- func max (n1, n2 int) int {
- var result int
- if n1 >= n2 {
- result = n1
- }
- if n1 < n2 {
- result = n2
- }
- return result
- }
- func main () {
- var result = max(5, 100)
- fmt.Println("max return", result)
- }
匿名函數(shù)
匿名函數(shù)就是一個(gè)沒有定義函數(shù)名的函數(shù),匿名函數(shù)可以當(dāng)成一個(gè)值,將其賦值放到某個(gè)變量中。這也是之前為什么說函數(shù)是『一等公民』,就是可以將函數(shù)當(dāng)成一個(gè)變量。
- var max = func (n1, n2 int) int {
- var result int
- if n1 >= n2 {
- result = n1
- }
- if n1 < n2 {
- result = n2
- }
- return result
- }
- var result = max(5, 100)
- fmt.Println("max return", result)
立即執(zhí)行函數(shù)
由于 go 中的函數(shù)是 『一等公民』,可以在聲明之后立即執(zhí)行,就是在函數(shù)聲明結(jié)束后,直接加上一個(gè)括號(hào),表示該函數(shù)會(huì)立即執(zhí)行,執(zhí)行之后的結(jié)果可以通過變量進(jìn)行接收。
- import "math"
- var Pi = func () float64 {
- return math.Pi
- }()
- fmt.Println("PI =",Pi)
閉包
“閉包就是能夠讀取其他函數(shù)內(nèi)部變量的函數(shù)。在本質(zhì)上,閉包是將函數(shù)內(nèi)部和函數(shù)外部連接起來的橋梁。 ——百度百科
上面的描述來自百度百科,初次看概念比較難理解,如果站在使用的角度來說,閉包就是在一個(gè)函數(shù)調(diào)用后,返回另一個(gè)匿名函數(shù),并保持當(dāng)前函數(shù)內(nèi)的局部變量,可以給匿名函數(shù)引用。
下面我們可以簡單實(shí)現(xiàn)一個(gè)迭代器函數(shù),該函數(shù)接受一個(gè)切片,返回一個(gè)匿名函數(shù),該匿名函數(shù)每次執(zhí)行,都會(huì)取出切片的一個(gè)值,直到全部讀取。
- func generate(slice []int) func() (bool, int) {
- i := 0
- length := len(slice)
- return func () (bool, int) {
- if i >= length {
- return true, 0
- }
- var result = slice[i]
- i++
- return false, result
- }
- }
- func main() {
- slice := []int{1, 2, 3, 4, 5}
- nextNum := generate(slice)
- done, result := nextNum()
- // 直到 done 不等于 false,才停止
- for done == false {
- fmt.Println(result, done)
- done, result = nextNum()
- }
- fmt.Println(result, done)
- }
指針
我們前面常說的變量指的一般是一個(gè)值,指針是指向該變量存儲(chǔ)在內(nèi)存的位置。指針也可以存儲(chǔ)在一個(gè)變量中,該變量稱為『指針變量』。
指針變量聲明
聲明指針變量時(shí),需要指針指向哪一種類型,因?yàn)椴煌愋偷闹翟趦?nèi)存占用的空間大小不一樣,僅僅知道內(nèi)存地址還是不夠,還需要知道該變量在內(nèi)存中占用多大空間。聲明指針變量只需要在類型前,加上 * 即可。
- var point *int // 聲明 int 類型的指針
指針變量賦值
給指針變量賦值,需要在對(duì)應(yīng)類型的變量前加上&符號(hào),表示取出該變量的地址。
- var i = 1
- var point *int
- point = &i
值傳遞與引用傳遞
一般情況下,我們傳入函數(shù)的參數(shù)僅為變量的值,這樣的傳遞被稱為值傳遞,在函數(shù)內(nèi)對(duì)參數(shù)修改也不會(huì)影響到外部變量。
- func addOne(slice []int, number int) {
- slice = append(slice, number)
- fmt.Println("inner slice =", slice)
- }
- slice := []int{1, 2, 3}
- addOne(slice, 100)
- fmt.Println("outer slice =", slice)
上述代碼中,我們寫了一個(gè)函數(shù),會(huì)對(duì)傳入的切片追加一個(gè)值,調(diào)用之后,我們會(huì)發(fā)現(xiàn)外部切片的值并沒有發(fā)生變量。
如果需要外部變量的值會(huì)跟隨函數(shù)調(diào)用發(fā)生變化,就需要將變量的指針傳入函數(shù)中,這樣的傳遞被稱為引用傳遞。這樣在函數(shù)中修改參數(shù)就會(huì)影響到外部的變量了。
- // 此時(shí) slice 為指針變量
- func addOne(slice *[]int, number int) {
- // 通過 *slice 可以取出 slice 指針對(duì)應(yīng)的值
- *slice = append(*slice, number)
- fmt.Println("inner slice =", *slice)
- }
- slice := []int{1, 2, 3}
- addOne(&slice, 100)
- fmt.Println("outer slice =", slice)