Go 面試系列(六) - err shadow 是什么鬼嚎杨?

在日常工作中,我們經(jīng)常使用 err != nil 來判斷程序或函數(shù)是否報(bào)錯(cuò)氧腰,或者使用 defer {recover = err} 來判斷是否有 panic 嚴(yán)重錯(cuò)誤枫浙,但稍不注意,很容易掉進(jìn) err shadow 的陷阱容贝。

1. 變量作用域

package main

import "fmt"

func main() {
    x := 100
    func() {
        x := 200 // x will shadow outer x
        fmt.Println(x)
    }()

    fmt.Println(x)
}

輸出如下:

200
100

結(jié)果分析:
x 變量在 func 里面打印為 200自脯,在外層打印為 100,這就是變量的作用域(variable scope)斤富。func 里面的變量 x 是一個(gè)新變量膏潮,只不過與外層 x 重名了(variable redeclaration),此時(shí)里層 x 的作用域僅限于 func {} block满力,而外層 x 的作用域則是 main {} block焕参,此時(shí)里層變量 x 發(fā)生了 variable shadowing,外層 x 不受影響油额,依然是 100叠纷。

改一下寫法:

package main

import "fmt"

func main() {
    x := 100
    func() {
        x = 200 // x will override outer x
        fmt.Println(x)
    }()

    fmt.Println(x)
}

輸出如下:

200
200

此時(shí),func 里面的變量 x 僅僅是覆蓋了外層 x潦嘶,并沒有定義新的變量涩嚣,所以內(nèi)外層輸出都是 200

2. err shadow - 無名 error

package main

import (
    "fmt"
    "os"
)

func main() {
    fmt.Println("func err1:", test1())
}

func test1() error {
    var err error

    defer func() {
        fmt.Println("defer err1:", err)
    }()

    if _, err := os.Open("xxx"); err != nil {
        return err
    }

    return nil
}

輸出如下:

defer err1: <nil>
func err1: open xxx: no such file or directory

結(jié)果分析:
func test1 首先定義了 var err error 變量掂僵,但下面的 os.Open 報(bào)錯(cuò)使用 err := 被局部 err shadow 了航厚,雖然顯式使用了 return err 返回錯(cuò)誤,但由于 test1() error 返回參數(shù)是無名的(unnamed variable)锰蓬,導(dǎo)致 defererr 獲取不到被 err shadow 的錯(cuò)誤 err幔睬,取的仍然是外層初始化 var err error 值,所以輸出為 err1: <nil>芹扭。

只需要將第 19 行改一下麻顶,即可避免 err shadow

if _, err = os.Open("xxx"); err != nil {
        return err
    }

輸出如下:

defer err1: open xxx: no such file or directory
func err1: open xxx: no such file or directory

3. err shadow - 有名 error

package main

import (
    "fmt"
    "os"
)

func main() {
    fmt.Println("func err2:", test2())
}

func test2() (err error) {

    defer func() {
        fmt.Println("defer err2:", err)
    }()

    if _, err := os.Open("xxx"); err != nil {
        return // return without err will compilation error
    }

    return
}

上面的 test2 運(yùn)行會(huì)有編譯報(bào)錯(cuò),這是 go compiler 在編譯時(shí)做了 variable shadowing 檢查舱卡,發(fā)現(xiàn)有就直接編譯報(bào)錯(cuò)辅肾。修改一下即可:

func main() {
    fmt.Println("func err3:", test3())
}

func test3() (err error) {

    defer func() {
        fmt.Println("defer err3:", err)
    }()

    if _, err := os.Open("xxx"); err != nil {
        return err
    }

    return
}

輸出如下:

defer err3: open xxx: no such file or directory
func err3: open xxx: no such file or directory

4. 嵌套 err shadow

package main

import (
    "encoding/json"
    "fmt"
    "os"
)

func main() {
    fmt.Println("func err4:", test4())
}

func test4() (err error) {

    defer func() {
        fmt.Println("defer err4:", err)
    }()

    if _, err := os.Open("xxx"); err == nil {
        if err := json.Unmarshal([]byte("{}"), &struct{}{}); err == nil {
            fmt.Println("OK")
        }
    }

    return
}

輸出如下:

defer err4: <nil>
func err4: <nil>

結(jié)果分析:
func test4() 是一個(gè)有名返回 err error,則函數(shù)初始化時(shí)會(huì) var err error 定義對(duì)應(yīng)的有名變量(named variable)轮锥,但下面的 os.Openjson.Unmarshal 都使用了 err := 重定義 err 變量宛瞄,造成了 err shadow,因此在函數(shù)退出時(shí),外層 err 依然是 nil份汗,defer 獲取也就是 nil盈电。

改一下寫法即可:

func main() {
    fmt.Println("func err5:", test5())
}

func test5() (err error) {

    defer func() {
        fmt.Println("defer err5:", err)
    }()

    if _, err = os.Open("xxx"); err == nil {
        if err = json.Unmarshal([]byte("{}"), &struct{}{}); err == nil {
            fmt.Println("OK")
        }
    }

    return
}

輸出如下:

defer err5: open xxx: no such file or directory
func err5: open xxx: no such file or directory

5. 小結(jié)

本文通過幾個(gè)實(shí)例,分析了在實(shí)際工作中很容易出現(xiàn)的 err shadow 問題杯活,究其本質(zhì)原因主要是變量作用域引起的匆帚,在官方文檔中提到:An identifier declared in a block may be redeclared in an inner block. While the identifier of the inner declaration is in scope, it denotes the entity declared by the inner declaration.

另外,在函數(shù)返回值命名方面旁钧,我們需要考慮無名吸重、有名參數(shù)的情況,在保證代碼邏輯正確的情況下歪今,建議使用工具 go lintergo vet 來檢測(cè)編譯器沒檢測(cè)到的 variable shadowing嚎幸,避免踩到坑。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末寄猩,一起剝皮案震驚了整個(gè)濱河市嫉晶,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌田篇,老刑警劉巖替废,帶你破解...
    沈念sama閱讀 206,482評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異泊柬,居然都是意外死亡椎镣,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,377評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門兽赁,熙熙樓的掌柜王于貴愁眉苦臉地迎上來状答,“玉大人,你說我怎么就攤上這事刀崖。” “怎么了蒲跨?”我有些...
    開封第一講書人閱讀 152,762評(píng)論 0 342
  • 文/不壞的土叔 我叫張陵授翻,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我堪唐,道長(zhǎng),這世上最難降的妖魔是什么淮菠? 我笑而不...
    開封第一講書人閱讀 55,273評(píng)論 1 279
  • 正文 為了忘掉前任合陵,我火速辦了婚禮枢赔,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘碎赢。我一直安慰自己,他們只是感情好肮塞,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,289評(píng)論 5 373
  • 文/花漫 我一把揭開白布姻锁。 她就那樣靜靜地躺著枕赵,像睡著了一般位隶。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上钓试,一...
    開封第一講書人閱讀 49,046評(píng)論 1 285
  • 那天,我揣著相機(jī)與錄音恋谭,去河邊找鬼挽鞠。 笑死疚颊,一個(gè)胖子當(dāng)著我的面吹牛信认,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播嫁赏,決...
    沈念sama閱讀 38,351評(píng)論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼款熬!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起攘乒,我...
    開封第一講書人閱讀 36,988評(píng)論 0 259
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎则酝,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,476評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡武鲁,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,948評(píng)論 2 324
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了洞坑。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,064評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡迟杂,死狀恐怖本慕,靈堂內(nèi)的尸體忽然破棺而出排拷,到底是詐尸還是另有隱情锅尘,我是刑警寧澤监氢,帶...
    沈念sama閱讀 33,712評(píng)論 4 323
  • 正文 年R本政府宣布藤违,位于F島的核電站,受9級(jí)特大地震影響顿乒,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜璧榄,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,261評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望涂身。 院中可真熱鬧,春花似錦蛤售、人聲如沸妒潭。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,264評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽拂盯。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間团驱,已是汗流浹背摸吠。 一陣腳步聲響...
    開封第一講書人閱讀 31,486評(píng)論 1 262
  • 我被黑心中介騙來泰國(guó)打工寸痢, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人啼止。 一個(gè)月前我還...
    沈念sama閱讀 45,511評(píng)論 2 354
  • 正文 我出身青樓兵罢,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親卖词。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,802評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容