<kbd id="afajh"><form id="afajh"></form></kbd>
<strong id="afajh"><dl id="afajh"></dl></strong>
    <del id="afajh"><form id="afajh"></form></del>
        1. <th id="afajh"><progress id="afajh"></progress></th>
          <b id="afajh"><abbr id="afajh"></abbr></b>
          <th id="afajh"><progress id="afajh"></progress></th>

          Go interface 類(lèi)型轉(zhuǎn)換原理剖析

          共 7165字,需瀏覽 15分鐘

           ·

          2021-08-13 19:57

          hi, 大家好,我是 haohongfan。

          可能你看過(guò)的 interface 剖析的文章比較多了,這些文章基本都是從匯編角度分析類(lèi)型轉(zhuǎn)換或者動(dòng)態(tài)轉(zhuǎn)發(fā)。不過(guò)隨著 Go 版本升級(jí),對(duì)應(yīng)的 Go 匯編也發(fā)生了巨大的變化,如果單從匯編角度去分析 interface 變的非常有難度,本篇文章我會(huì)從內(nèi)存分配+匯編角度切入 interface,去了解 interface 的原理。

          限于篇幅 interface 有關(guān)動(dòng)態(tài)轉(zhuǎn)發(fā)和反射的內(nèi)容,請(qǐng)關(guān)注后續(xù)的文章。本篇文章主要是關(guān)于類(lèi)型轉(zhuǎn)換。

          eface
          iface

          eface

          func main() {
             var ti interface{}
             var a int = 100
             ti = a
             fmt.Println(ti)
          }

          這段最常見(jiàn)的代碼,現(xiàn)在提出一些問(wèn)題:

          • 如何查看 ti 是 eface 還是 iface ?
          • 值 100 保存在哪里了 ?
          • 如何看 ti 的真實(shí)的值的類(lèi)型 ?

          大部分源碼分析都是從匯編入手來(lái)看的,這里也把對(duì)應(yīng)的匯編貼出來(lái)

          0x0040 00064 (main.go:44)	MOVQ	$100, (SP)
          0x0048 00072 (main.go:44) CALL runtime.convT64(SB)
          0x004d 00077 (main.go:44) MOVQ 8(SP), AX
          0x0052 00082 (main.go:44) MOVQ AX, ""..autotmp_3+64(SP)
          0x0057 00087 (main.go:44) LEAQ type.int(SB), CX
          0x005e 00094 (main.go:44) MOVQ CX, "".ti+72(SP)
          0x0063 00099 (main.go:44) MOVQ AX, "".ti+80(SP)

          這段匯編有下面這些特點(diǎn):

          • CALL runtime.convT64(SB):將 100 作為 runtime.convT64 的參數(shù),該函數(shù)申請(qǐng)了一段內(nèi)存,將 100 放入了這段內(nèi)存里
          • 將類(lèi)型 type.int 放入到 SP+72 的位置
          • 將包含 100 的那塊內(nèi)存的指針,放入到 SP + 80 的位置

          這段匯編從直觀上來(lái)說(shuō),interface 轉(zhuǎn)換成 eface 是看不出來(lái)的。這個(gè)如何觀察呢?這個(gè)就需要借助 gdb 了。

          再繼續(xù)深究下,如何利用內(nèi)存分布來(lái)驗(yàn)證是 eface 呢?需要另外再添加點(diǎn)代碼。

          type eface struct {
              _type *_type
              data  unsafe.Pointer
          }

          type _type struct {
              size       uintptr
              ptrdata    uintptr // size of memory prefix holding all pointers
              hash       uint32
              tflag      tflag
              align      uint8
              fieldAlign uint8
              kind       uint8
              equal      func(unsafe.Pointer, unsafe.Pointer) bool
              gcdata     *byte
              str        nameOff
              ptrToThis  typeOff
          }

          func main() {
              var ti interface{}
              var a int = 100
              ti = a

              fmt.Println("type:", *(*eface)(unsafe.Pointer(&ti))._type)
              fmt.Println("data:", *(*int)((*eface)(unsafe.Pointer(&ti)).data))
              fmt.Println((*eface)(unsafe.Pointer(&ti)))
          }

          output:

          type: {8 0 4149441018 15 8 8 2 0x10032e0 0x10e6b60 959 27232}
          data: 100
          &{0x10ade20 0x1155bc0}

          從這個(gè)結(jié)果上能夠看出來(lái)

          • eface.kind = 2, 對(duì)應(yīng)著 runtime.kindInt
          • eface.data = 100

          從內(nèi)存上分配上看,我們基本看出來(lái)了 eface 的內(nèi)存布局及對(duì)應(yīng)的最終的 eface 的類(lèi)型轉(zhuǎn)換結(jié)果。

          iface

          package main

          type Person interface {
             Say() string
          }

          type Man struct {
          }

          func (m *Man) Say() string {
             return "Man"
          }

          func main() {
              var p Person

              m := &Man{}
              p = m
              println(p.Say())
          }

          iface 我們也看下匯編:

          0x0029 00041 (main.go:24)	LEAQ	runtime.zerobase(SB), AX
          0x0030 00048 (main.go:24) MOVQ AX, ""..autotmp_6+48(SP)
          0x0035 00053 (main.go:24) MOVQ AX, "".m+32(SP)
          0x003a 00058 (main.go:25) MOVQ AX, ""..autotmp_3+64(SP)
          0x003f 00063 (main.go:25) LEAQ go.itab.*"".Man,"".Person(SB), CX
          0x0046 00070 (main.go:25) MOVQ CX, "".p+72(SP)
          0x004b 00075 (main.go:25) MOVQ AX, "".p+80(SP)

          這段匯編上,能夠看出來(lái)是有 itab 的,但是是否真的是轉(zhuǎn)成了 iface,匯編上仍然反應(yīng)不出來(lái)。

          同樣,我們繼續(xù)用 gdb 查看 Person interface 確實(shí)被轉(zhuǎn)換成了 iface。

          關(guān)于 iface 內(nèi)存布局,我們?nèi)匀患狱c(diǎn)代碼來(lái)查看

          type itab struct {
              inter *interfacetype
              _type *_type
              hash  uint32
              _     [4]byte
              fun   [1]uintptr
          }

          type iface struct {
              tab  *itab
              data unsafe.Pointer
          }

          type Person interface {
              Say() string
          }

          type Man struct {
              Name string
          }

          func (m *Man) Say() string {
              return "Man"
          }

          func main() {
              var p Person

              m := &Man{Name: "hhf"}
              p = m
              println(p.Say())

              fmt.Println("itab:", *(*iface)(unsafe.Pointer(&p)).tab)
              fmt.Println("data:", *(*Man)((*iface)(unsafe.Pointer(&p)).data))
          }

          output:

          Man
          itab: {0x10b3ba0 0x10b1900 1224794265 [0 0 0 0] [17445152]}
          data: {hhf}

          關(guān)于想繼續(xù)探究 eface, iface 的內(nèi)存布局的同學(xué),可以基于上面的代碼,利用 unsafe 的相關(guān)函數(shù)去看對(duì)應(yīng)的內(nèi)存位置上的值。

          類(lèi)型斷言

          type Person interface {
             Say() string
          }

          type Man struct {
             Name string
          }

          func (m *Man) Say() string {
             return "Man"
          }

          func main() {
             var p Person

              m := &Man{Name: "hhf"}
              p = m

              if m1, ok := p.(*Man); ok {
                fmt.Println(m1.Name)
              }
          }

          我們僅關(guān)注類(lèi)型斷言那塊內(nèi)容,貼出對(duì)應(yīng)的匯編

          0x0087 00135 (main.go:23)	MOVQ	"".p+104(SP), AX
          0x008c 00140 (main.go:23) MOVQ "".p+112(SP), CX
          0x0091 00145 (main.go:23) LEAQ go.itab.*"".Man,"".Person(SB), DX
          0x0098 00152 (main.go:23) CMPQ DX, AX

          能夠看出來(lái)的是:將 iface.itab 放入了 AX,將 go.itab.*"".Man,"".Person(SB) 放入了 DX,比較兩者是否相等,來(lái)判斷 Person 的真實(shí)類(lèi)型是否是 Man。

          另外一個(gè)類(lèi)型斷言的方式就是 switch 了,其實(shí)兩者本質(zhì)上沒(méi)啥區(qū)別。

          interface 最著名的坑的,應(yīng)該就是下面這個(gè)了。

          func main() {
              var a interface{} = nil
              var b *int = nil
              
              isNil(a)
              isNil(b)
          }

          func isNil(x interface{}) {
              if x == nil {
                fmt.Println("empty interface")
                return
              }
              fmt.Println("non-empty interface")
          }

          output:

          empty interface
          non-empty interface

          為什么會(huì)這樣呢?這就涉及到 interface == nil 的判斷方式了。一般情況只有 eface 的 type 和 data 都為 nil 時(shí),interface == nil 才是 true。

          當(dāng)我們把 b 復(fù)制給 interface 時(shí),x._type.Kind = kindPtr。雖說(shuō) x.data = nil,但是不符合 interface == nil 的判斷條件了。

          關(guān)于 interface 源碼閱讀的一點(diǎn)建議

          關(guān)于 interface 源碼閱讀的一點(diǎn)建議,如果想利用匯編看源碼的話,盡量選擇 go1.14.x。

          選擇 Go 匯編來(lái)看 interface,基本上也是為了查看 interface 最終被轉(zhuǎn)換成 eface 還是 iface,調(diào)用了 runtime 的哪些函數(shù),以及對(duì)應(yīng)的函數(shù)棧分布。如果 Go 版本選擇的太高的話,go 匯編變化太大了,可能匯編上就看不到對(duì)應(yīng)的內(nèi)容了。

          歡迎關(guān)注公眾號(hào)。更多學(xué)習(xí)學(xué)習(xí)資料分享,關(guān)注公眾號(hào)回復(fù)指令:

          • 回復(fù) 0,獲取 《Go 面經(jīng)》

          • 回復(fù) 1,獲取 《Go 源碼流程圖》



          瀏覽 52
          點(diǎn)贊
          評(píng)論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報(bào)
          評(píng)論
          圖片
          表情
          推薦
          點(diǎn)贊
          評(píng)論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報(bào)
          <kbd id="afajh"><form id="afajh"></form></kbd>
          <strong id="afajh"><dl id="afajh"></dl></strong>
            <del id="afajh"><form id="afajh"></form></del>
                1. <th id="afajh"><progress id="afajh"></progress></th>
                  <b id="afajh"><abbr id="afajh"></abbr></b>
                  <th id="afajh"><progress id="afajh"></progress></th>
                  国产精品一二 | 色女人综合 | 日本欧美在线视频 | 蜜桃无码网站 | 成 人 网 站 观看 |