<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>

          這個(gè) Go 語(yǔ)言的經(jīng)典 “坑”,我算是服了

          共 2915字,需瀏覽 6分鐘

           ·

          2022-04-30 00:16

          在開(kāi)始之前,先考你一個(gè)非常 Go 味的經(jīng)典問(wèn)題:如何判斷一個(gè) interface{} 的值是否為 nil ?

          這也是面試有可能會(huì)被問(wèn)到的一個(gè)問(wèn)題,這個(gè)問(wèn)題很 “迷”,平時(shí)沒(méi)有特別留心的朋友,很容易在這邊裁了。

          我相信很多人會(huì)下意識(shí)的回答,直接 v == nil 進(jìn)行判斷不就好了嗎?

          很久之前,我也是那么想的,可寫(xiě)了個(gè) demo 后,才發(fā)現(xiàn)事情沒(méi)那么簡(jiǎn)單。

          請(qǐng)看下面這段代碼,可以先猜測(cè)一下輸出的結(jié)果。

          package?main

          import?(
          ????"fmt"
          )

          func?main()??{
          ????var?a?*string?=?nil
          ????var?b?interface{}?=?a

          ????fmt.Println(b==nil)?
          }

          答案應(yīng)該會(huì)跟你下意識(shí)的回答 相反

          輸出的結(jié)果的是 false

          這是為什么呢?接下來(lái),我們就要好好的嘮嘮這里面的道道。

          # 1. 兩個(gè) interface 比較

          interface 的內(nèi)部實(shí)現(xiàn)包含了兩個(gè)字段,一個(gè)是 type,一個(gè)是 data

          對(duì)于這樣一個(gè)變量

          var?age?interface{}?=?25

          其實(shí)內(nèi)部結(jié)構(gòu)是

          因此兩個(gè) interface 比較,勢(shì)必與這兩個(gè)字段有所關(guān)系。

          經(jīng)過(guò)驗(yàn)證,只有下面兩種情況,兩個(gè) interface 才會(huì)相等。

          ?第一種情況

          type 和 data 都相等

          在下面的代碼中,p1 和 p2 的 type 都是 Profile,data 都是 {"iswbm"},因此 p1 與 p2 相等

          而 p3 和 p3 雖然類(lèi)型都是 *Profile,但由于 data 存儲(chǔ)的是結(jié)構(gòu)體的地址,而兩個(gè)地址和不相同,因此 p3 與 p4 不相等

          package?main

          import?"fmt"

          type?Profile?struct?{
          ????Name?string
          }

          type?ProfileInt?interface?{}

          func?main()??{
          ????var?p1,?p2?ProfileInt?=?Profile{"iswbm"},?Profile{"iswbm"}
          ????var?p3,?p4?ProfileInt?=?&Profile{"iswbm"},?&Profile{"iswbm"}

          ????fmt.Printf("p1?-->?type:?%T,?data:?%v?\n",?p1,?p1)
          ????fmt.Printf("p2?-->?type:?%T,?data:?%v?\n",?p2,?p2)
          ????fmt.Println(p1?==?p2)??//?true

          ????fmt.Printf("p3?-->?type:?%T,?data:?%p?\n",?p3,?p3)
          ????fmt.Printf("p4?-->?type:?%T,?data:?%p?\n",?p4,?p4)
          ????fmt.Println(p3?==?p4)??//?false
          }

          運(yùn)行后,輸出如下

          p1?-->?type:?main.Profile,?data:?{iswbm}?
          p2?-->?type:?main.Profile,?data:?{iswbm}?
          true
          p3?-->?type:?*main.Profile,?data:?0xc00008e200?
          p4?-->?type:?*main.Profile,?data:?0xc00008e210?
          false

          ?第二種情況

          特殊情況:兩個(gè) interface 都是 nil

          當(dāng)一個(gè) interface 的 type 和 data 都處于 unset 狀態(tài)的時(shí)候,那么該 interface 的類(lèi)型和值都為 nil

          package?main

          import?"fmt"

          func?main()?{
          ????var?p1,?p2?interface{}
          ????fmt.Println(p1?==?p2)??//?true
          ????fmt.Println(p1?==?nil)?//?true
          }

          # 2. interface 與 非 interface 比較

          當(dāng) interface 與非 interface 比較時(shí),會(huì)將 非interface 轉(zhuǎn)換成 interface ,然后再按照 兩個(gè) interface 比較 的規(guī)則進(jìn)行比較。

          示例如下

          package?main

          import?(
          ????"fmt"
          ????"reflect"
          )

          func?main()??{
          ????var?a?string?=?"iswbm"
          ????var?b?interface{}?=?"iswbm"
          ????fmt.Println(a==b)?//?true
          }

          上面這種例子可能還好理解,那么請(qǐng)你看下面這個(gè)例子(文章開(kāi)頭的示例),為什么 b 與 nil 不相等?

          package?main

          import?(
          ????"fmt"
          )

          func?main()??{
          ????var?a?*string?=?nil
          ????var?b?interface{}?=?a

          ????fmt.Println(b==nil)?//?false
          }

          但當(dāng)你使用 b==nil 進(jìn)行判斷時(shí),其實(shí)右邊的 nil 并非單純的是我們所理解的值為nil,而正確的理解應(yīng)該是 類(lèi)型為 nil 且 值也為 nil。

          Go 會(huì)先將 nil 轉(zhuǎn)換為interface ?(type=nil, data=nil) ,這與 b (type=*string, data=nil) ?雖然 data 是一樣的,但 type 不相等,因此他們并不相等。

          那有沒(méi)有辦法判斷一個(gè) interface{} 是不是 nil 呢?

          有辦法的,但是要借助反射,一個(gè)非萬(wàn)不得已不會(huì)使用的 reflect 包。

          package?main

          import?(
          ????"fmt"
          ????"reflect"
          )

          func?IsNil(i?interface{})?bool?{
          ????vi?:=?reflect.ValueOf(i)
          ????if?vi.Kind()?==?reflect.Ptr?{
          ????????return?vi.IsNil()
          ????}
          ????return?false
          }

          func?main()?{
          ????var?a?*string?=?nil
          ????var?b?interface{}?=?a
          ????fmt.Println(IsNil(b))
          }


          本文通過(guò)一些例子介紹了 Go 在比較時(shí)候,內(nèi)部的一些實(shí)現(xiàn)原理,對(duì)于很多人,可能是一個(gè)“新”知識(shí),沒(méi)有掌握的話(huà),一定會(huì)在以后在編碼過(guò)程中給自己挖了個(gè)“大坑”,而這種語(yǔ)言?xún)?nèi)部的 “坑”,不知道就是不知道,再怎么代碼走查都很難發(fā)現(xiàn)。希望通過(guò)本篇文章,帶你一起把這個(gè)“坑” 給填上。


          本篇原屬于 Go 面試題庫(kù)專(zhuān)欄系列文章,以前都是在標(biāo)題上寫(xiě)明了是面試題,考慮到有些人最近沒(méi)有在面試,怕你們錯(cuò)過(guò)這類(lèi)即使不面試,也要掌握的知識(shí),以后的內(nèi)容,可能不會(huì)在標(biāo)題上特別標(biāo)明是面試題了。

          本專(zhuān)欄系列文章,我都公開(kāi)到 Github 上:https://github.com/iswbm/golang-interview

          瀏覽 46
          點(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>
                  正在播放小早川无码AV | 久久久91人妻无码精品蜜桃ID | 成人艹逼视频 | 成人视频毛片 | 亚洲人成人网站色 |