go中panic以及recover使用

背景

上周五快下班的時(shí)候蝇刀,突然客戶報(bào)服務(wù)出現(xiàn)了503 http錯(cuò)誤譬淳,既服務(wù)不可用募舟。而且這個(gè)錯(cuò)誤是用戶在進(jìn)行某個(gè)操作的時(shí)候出現(xiàn)的祠斧。當(dāng)時(shí)去看了客戶那邊的日志,沒(méi)有發(fā)現(xiàn)有價(jià)值的信息拱礁。好在我們知道是哪個(gè)操作導(dǎo)致了這個(gè)問(wèn)題琢锋,將客戶數(shù)據(jù)拿到,然后在本地進(jìn)行測(cè)試觅彰,發(fā)現(xiàn)果然出現(xiàn)了panic,導(dǎo)致服務(wù)重啟了钮热。
經(jīng)過(guò)排查填抬,發(fā)現(xiàn)是因?yàn)榇a中出現(xiàn)了越界訪問(wèn)。但是有個(gè)問(wèn)題一直困擾隧期,為什么代碼中寫(xiě)了gin panic recover代碼飒责,但是這段代碼并沒(méi)有生效呢?帶著這個(gè)疑問(wèn)仆潮,做了響應(yīng)的實(shí)驗(yàn)宏蛉,發(fā)現(xiàn)對(duì)于go中panic理解還是沒(méi)有到位。
總結(jié)下來(lái)性置,go中panic的recover有以下兩個(gè)原則

原則1:如果程序中出現(xiàn)了panic拾并,而沒(méi)有進(jìn)行recover,那么程序會(huì)直接退出

原則2:一個(gè)協(xié)程中出現(xiàn)的panic,在另一個(gè)協(xié)程中是無(wú)法recover的嗅义,只能在本協(xié)程中進(jìn)行recover

原則3:go中某些panic是無(wú)法進(jìn)行recover的屏歹,比如并發(fā)讀寫(xiě)map(非線程安全的map結(jié)構(gòu))

下面分別對(duì)于這三個(gè)原則,編寫(xiě)相關(guān)的代碼進(jìn)行驗(yàn)證

原則1:如果程序中出現(xiàn)了panic之碗,而沒(méi)有進(jìn)行recover蝙眶,那么程序會(huì)直接退出

package main

import "fmt"

func main() {
    panic("error")

    fmt.Println("hello world")
}

輸出結(jié)果如下:

panic: error

goroutine 1 [running]:
main.main()
        .../go/src/awesomeProject/demo1/main.go:6 +0x2c
exit status 2

可以看到hello world并沒(méi)有打印出來(lái),程序直接退出了

原則2:一個(gè)協(xié)程中出現(xiàn)的panic褪那,在另一個(gè)協(xié)程中是無(wú)法recover的幽纷,只能在本協(xié)程中進(jìn)行recover

例子1:main函數(shù)中使用recover

package main

import "fmt"

func main() {
    defer func() {
        fmt.Println("enter defer")
        if err := recover(); err != nil {
            fmt.Println(err)
        }
    }()
    nums := []int{1, 2, 3}
    var index = 4
    fmt.Println(nums[index])
    fmt.Println("done")
}

上面這段程序輸出結(jié)果如下:

enter defer
runtime error: index out of range [4] with length 3

可以看到在main函數(shù)中使用recover并沒(méi)有拯救程序,程序仍然是退出的博敬。

例子2:
但是下面這段程序就可以

package main

import (
    "fmt"
)

func main() {
    f()
    fmt.Println("hello world")
}

func f() {
    defer func() {
        fmt.Println("enter defer")
        if err := recover(); err != nil {
            fmt.Println(err)
        }
    }()
    nums := []int{1, 2, 3}
    var index = 4
    fmt.Println(nums[index])
}

輸出

enter defer
runtime error: index out of range [4] with length 3
hello world

例子3:在另一個(gè)協(xié)程中recover

package main

import (
    "fmt"
    "time"
)

func main() {
    go func() {
        if e := recover(); e != nil {
            fmt.Println(e)
        }
    }()
    go func() {
        nums := []int{1, 2, 3}
        var index = 4
        fmt.Println(nums[index])
    }()
    time.Sleep(1 * time.Second)
    fmt.Println("done")
}

輸出結(jié)果

panic: runtime error: index out of range [4] with length 3

goroutine 19 [running]:
main.main.func2()
       ..../go/src/awesomeProject/demo3/main.go:17 +0x24
created by main.main
       .../go/src/awesomeProject/demo3/main.go:14 +0x30
exit status 2
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末友浸,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子冶忱,更是在濱河造成了極大的恐慌尾菇,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,372評(píng)論 6 498
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件囚枪,死亡現(xiàn)場(chǎng)離奇詭異派诬,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)链沼,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,368評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門(mén)默赂,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人括勺,你說(shuō)我怎么就攤上這事缆八。” “怎么了疾捍?”我有些...
    開(kāi)封第一講書(shū)人閱讀 162,415評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵奈辰,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我乱豆,道長(zhǎng)奖恰,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,157評(píng)論 1 292
  • 正文 為了忘掉前任宛裕,我火速辦了婚禮瑟啃,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘揩尸。我一直安慰自己蛹屿,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,171評(píng)論 6 388
  • 文/花漫 我一把揭開(kāi)白布岩榆。 她就那樣靜靜地躺著错负,像睡著了一般坟瓢。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上湿颅,一...
    開(kāi)封第一講書(shū)人閱讀 51,125評(píng)論 1 297
  • 那天载绿,我揣著相機(jī)與錄音,去河邊找鬼油航。 笑死崭庸,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的谊囚。 我是一名探鬼主播怕享,決...
    沈念sama閱讀 40,028評(píng)論 3 417
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼镰踏!你這毒婦竟也來(lái)了函筋?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 38,887評(píng)論 0 274
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤奠伪,失蹤者是張志新(化名)和其女友劉穎跌帐,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體绊率,經(jīng)...
    沈念sama閱讀 45,310評(píng)論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡谨敛,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,533評(píng)論 2 332
  • 正文 我和宋清朗相戀三年眶熬,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了宜猜。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,690評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡缴渊,死狀恐怖藐俺,靈堂內(nèi)的尸體忽然破棺而出炊甲,到底是詐尸還是另有隱情,我是刑警寧澤欲芹,帶...
    沈念sama閱讀 35,411評(píng)論 5 343
  • 正文 年R本政府宣布卿啡,位于F島的核電站,受9級(jí)特大地震影響菱父,放射性物質(zhì)發(fā)生泄漏颈娜。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,004評(píng)論 3 325
  • 文/蒙蒙 一滞伟、第九天 我趴在偏房一處隱蔽的房頂上張望揭鳞。 院中可真熱鬧炕贵,春花似錦梆奈、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,659評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)乓梨。三九已至,卻和暖如春清酥,著一層夾襖步出監(jiān)牢的瞬間扶镀,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,812評(píng)論 1 268
  • 我被黑心中介騙來(lái)泰國(guó)打工焰轻, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留臭觉,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 47,693評(píng)論 2 368
  • 正文 我出身青樓辱志,卻偏偏與公主長(zhǎng)得像蝠筑,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子揩懒,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,577評(píng)論 2 353

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