佳木斯湛栽影视文化发展公司

主頁 > 知識庫 > 深入理解Go語言中的閉包

深入理解Go語言中的閉包

熱門標簽:地方門戶網(wǎng)站 Linux服務器 網(wǎng)站排名優(yōu)化 鐵路電話系統(tǒng) 服務外包 AI電銷 呼叫中心市場需求 百度競價排名

閉包

在函數(shù)編程中經(jīng)常用到閉包,閉包是什?它是怎么產(chǎn)生的及用來解決什么問題呢?先給出閉包的字面定義:閉包是由函數(shù)及其相關引用環(huán)境組合而成的實體(即:閉包=函數(shù)+引用環(huán)境)。這個從字面上很難理解,特別對于一直使用命令式語言進行編程的程序員們。

Go語言中的閉包

先看一個demo:

func f(i int) func() int {
 return func() int {
 i++
 return i
 }
}

函數(shù)f返回了一個函數(shù),返回的這個函數(shù)就是一個閉包。這個函數(shù)中本身是沒有定義變量i的,而是引用了它所在的環(huán)境(函數(shù)f)中的變量i。

我們再看一下效果:

c1 := f(0)
c2 := f(0)
c1() // reference to i, i = 0, return 1
c2() // reference to another i, i = 0, return 1

c1跟c2引用的是不同的環(huán)境,在調(diào)用i++時修改的不是同一個i,因此兩次的輸出都是1。函數(shù)f每進入一次,就形成了一個新的環(huán)境,對應的閉包中,函數(shù)都是同一個函數(shù),環(huán)境卻是引用不同的環(huán)境。

變量i是函數(shù)f中的局部變量,假設這個變量是在函數(shù)f的棧中分配的,是不可以的。因為函數(shù)f返回以后,對應的棧就失效了,f返回的那個函數(shù)中變量i就引用一個失效的位置了。所以閉包的環(huán)境中引用的變量不能夠在棧上分配。

escape analyze

在繼續(xù)研究閉包的實現(xiàn)之前,先看一看Go的一個語言特性:

func f() *Cursor {
 var c Cursor
 c.X = 500
 noinline()
 return c
}

Cursor是一個結構體,這種寫法在C語言中是不允許的,因為變量c是在棧上分配的,當函數(shù)f返回后c的空間就失效了。但是,在Go語言規(guī)范中有說明,這種寫法在Go語言中合法的。語言會自動地識別出這種情況并在堆上分配c的內(nèi)存,而不是函數(shù)f的棧上。

為了驗證這一點,可以觀察函數(shù)f生成的匯編代碼:

MOVQ $type."".Cursor+0(SB),(SP) // 取變量c的類型,也就是Cursor
PCDATA $0,$16
PCDATA $1,$0
CALL ,runtime.new(SB) // 調(diào)用new函數(shù),相當于new(Cursor)
PCDATA $0,$-1
MOVQ 8(SP),AX // 取c.X的地址放到AX寄存器
MOVQ $500,(AX) // 將AX存放的內(nèi)存地址的值賦為500
MOVQ AX,"".~r0+24(FP)
ADDQ $16,SP

識別出變量需要在堆上分配,是由編譯器的一種叫escape analyze的技術實現(xiàn)的。

如果輸入命令:

go build --gcflags=-m main.go

可以看到輸出:

注意:最后兩行,標識c逃逸了,被移動到堆中。escape analyze可以分析出變量的作用范圍,這是對垃圾回收很重要的一項技術。

總結

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作能帶來一定的幫助,如果有疑問大家可以留言交流。

您可能感興趣的文章:
  • GO語io包的常用接口
  • Go語言的os包中常用函數(shù)初步歸納
  • 舉例講解Go語言中函數(shù)的閉包使用
  • go語言里包的用法實例
  • Go語言聲明一個多行字符串的變量
  • Go語言基礎知識總結(語法、變量、數(shù)值類型、表達式、控制結構等)
  • GO語言獲取系統(tǒng)環(huán)境變量的方法
  • Go語言中的變量聲明和賦值
  • 詳解Golang編程中的常量與變量
  • go語言學習之包和變量詳解

標簽:衡水 蘭州 仙桃 黃山 湖南 銅川 湘潭 崇左

巨人網(wǎng)絡通訊聲明:本文標題《深入理解Go語言中的閉包》,本文關鍵詞  ;如發(fā)現(xiàn)本文內(nèi)容存在版權問題,煩請?zhí)峁┫嚓P信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡,涉及言論、版權與本站無關。
  • 相關文章
  • 收縮
    • 微信客服
    • 微信二維碼
    • 電話咨詢

    • 400-1100-266
    英山县| 襄城县| 汕尾市| 茌平县| 朝阳区| 湟中县| 醴陵市| 塔河县| 阜平县| 浦城县| 岫岩| 积石山| 固镇县| 邛崃市| 洛南县| 福海县| 浑源县| 贵阳市| 那曲县| 汉中市| 永靖县| 合阳县| 石门县| 炎陵县| 上思县| 辽阳县| 左权县| 富平县| 诏安县| 汤原县| 宜章县| 昌平区| 独山县| 黔东| 青冈县| 泗水县| 大化| 张家川| 旬阳县| 北流市| 宜黄县|