亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > Golang > 正文

Golang巧用defer進行錯誤處理的方法

2020-04-01 19:05:38
字體:
來源:轉載
供稿:網友

本文主要跟大家介紹了Golang巧用golang/196851.html">defer進行錯誤處理的相關內容,分享出來供大家參考學習,下面來看看詳細的介紹:

問題引入

毫無疑問,錯誤處理是程序的重要組成部分,有效且優雅的處理錯誤是大多數程序員的追求。很多程序員都有C/C++的編程背景,Golang的程序員也不例外,他們處理錯誤有意無意的帶著C/C++的烙印。

我們看看下面的例子,就有一種似曾相識的趕腳,代碼如下:

func deferDemo() error { err := createResource1() if err != nil { return ERR_CREATE_RESOURCE1_FAILED } err = createResource2() if err != nil { destroyResource1() return ERR_CREATE_RESOURCE2_FAILED } err = createResource3() if err != nil { destroyResource1() destroyResource2() return ERR_CREATE_RESOURCE3_FAILED } err = createResource4() if err != nil { destroyResource1() destroyResource2() destroyResource3() return ERR_CREATE_RESOURCE4_FAILED } return nil}

從代碼的實現中可以看出:在一個函數中,當創建新資源失敗時,則要清理所有前面已經創建成功的資源,這使得函數中有了重復代碼的壞味道,比如destroyResource1函數調用了3次,destroyResource2函數調用了2次。

重構一:一個defer + 多個flag

Golang提供了一個很好用的關鍵字defer,當包含defer的函數執行完畢時(不管是通過return的正常結束,還是由于panic導致的異常結束),defer語句才被調用。

考慮到這一點,我們嘗試將所有資源在defer語句中統一清理。由于函數返回時,不知道是否需要清理以及清理那些資源,所以要增加多個flag。

重構后的代碼如下所示:

func deferDemo() error { flag := false flag1 := false flag2 := false flag3 := false defer func() { if !flag { if flag3 {  destroyResource3() } if flag2 {  destroyResource2() } if flag1 { destroyResource1() } } }() err := createResource1() if err != nil { return ERR_CREATE_RESOURCE1_FAILED } flag1 = true err = createResource2() if err != nil { return ERR_CREATE_RESOURCE2_FAILED } flag2 = true err = createResource3() if err != nil { return ERR_CREATE_RESOURCE3_FAILED } flag3 = true err = createResource4() if err != nil { return ERR_CREATE_RESOURCE4_FAILED } flag = true return nil}

從重構后的代碼可以看出,雖然消除了重復,但是引入了太多的flag:

  • flag表示函數是否執行成功,即flag為true時表示函數執行成功,否則表示函數執行失??;在defer語句中,只有flag為false時才需要統一清理資源
  • flagi表示第i個資源是否創建成功,即flagi為true時表示第i個資源創建成功,否則表示第i個資源創建失敗;在defer語句中,只有flagi為true時才需要清理第i個資源

顯然,這不是我們想要的

重構二:多個defer

看過linux源碼的同學都知道,在內核代碼中,很多地方都通過goto語句來集中處理錯誤,非常優雅。

我們用這種方法將重構前的代碼用C語言寫一下,代碼如下所示:

ErrCode deferDemo(){ ErrCode err = createResource1(); if (err != ERR_SUCC) { goto err_1; } err = createResource2(); if (err != ERR_SUCC) { goto err_2; } err = createResource3(); if (err != ERR_SUCC) { goto err_3; } err = createResource4(); if (err != ERR_SUCC) { goto err_4; } return ERR_SUCC; err_4: destroyResource3(); err_3: destroyResource2(); err_2: destroyResource1(); err_1: return ERR_FAIL;}

沒有重復,沒有flag,錯誤處理也很優雅,感覺很爽,那以前在C/C++編碼規范中禁止使用goto語句的規則確實有點過,呵呵...

從重構后的C代碼中可以看出,create操作和destroy操作的順序類似入棧和出棧的順序:

  • 伴隨著create操作,destroy操作逐個入棧,順序為1,2,3
  • 出棧時是destroy操作,順序為3,2,1

于是我們又想到了defer語句:當Golang的代碼執行時,如果遇到defer語句,則壓入堆棧,當函數返回時,會按照后進先出的順序調用defer語句。

我們看一個例子,代碼如下所示:

func main() { defer fmt.Println(1) defer fmt.Println(2) defer fmt.Println(3)}

運行后,日志如下所示:

321

然而,有堆棧特性還不夠,因為伴隨著create操作,destroy操作入棧是有條件的:

  • 如果create操作失敗,則直接返回,那么defer語句沒有執行,導致destroy操作沒有入棧
  • 如果create操作成功,則defer語句得到執行,destroy操作完成入棧

可見,destroy操作的入棧條件是create操作成功,但是destroy操作并不是一定執行,只有當某個create操作失敗("err != nil")時,前面入棧的destory操作才需要執行,所以err的值也需要入棧。然而,destroy操作入棧時"err == nil" ,于是問題就變成:當err的值在后面變成非nil時,應該同步修改堆棧中的err值,即堆棧中傳遞的是引用或指針而不是值。

當err的引用或指針和destroy操作都需要入棧時,defer后面必須是一個閉包調用。我們知道,對于閉包的參數是值傳遞,而對于外部變量卻是引用傳遞。為了簡單優雅起見,我們將err不通過參數的指針傳遞,而通過外部變量的引用傳遞。

我們根據這個結論重構一下代碼,如下所示:

func deferDemo() error { err := createResource1() if err != nil { return ERR_CREATE_RESOURCE1_FAILED } defer func() { if err != nil { destroyResource1() } }() err = createResource2() if err != nil { return ERR_CREATE_RESOURCE2_FAILED } defer func() { if err != nil { destroyResource2() } }() err = createResource3() if err != nil { return ERR_CREATE_RESOURCE3_FAILED } defer func() { if err != nil { destroyResource3() } }() err = createResource4() if err != nil { return ERR_CREATE_RESOURCE4_FAILED } return nil}

本次重構消除了代碼的壞味道,不由的感嘆一句:”升級了,我的哥!“

總結

本文通過巧用defer,有效且優雅的處理了錯誤,該技巧應該被所有的Golang程序員掌握并大量使用。

好了,以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作能帶來一定的幫助,如果有疑問大家可以留言交流,謝謝大家對VEVB武林網的支持。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
精品久久久久久亚洲国产300| 亚洲第一区在线| 精品久久久久久久大神国产| 国产视频自拍一区| 姬川优奈aav一区二区| 日本精品一区二区三区在线播放视频| 成人黄色在线播放| 亚洲国产美女久久久久| 久久久久久国产精品美女| 国产精品久久久久久av福利软件| 亚洲综合最新在线| 欧美日韩福利在线观看| 51久久精品夜色国产麻豆| 亚洲欧洲偷拍精品| 夜夜狂射影院欧美极品| 久久久97精品| 亚洲天堂av图片| 1769国内精品视频在线播放| 色偷偷亚洲男人天堂| 91在线观看免费高清完整版在线观看| 91久久在线播放| 日韩av中文字幕在线| 视频一区视频二区国产精品| 欧美xxxx14xxxxx性爽| 国产欧美韩国高清| 国产精品视频一| 欧美日本在线视频中文字字幕| 亚洲图片在区色| 亚洲成人中文字幕| 国产精品久久久精品| 91亚洲永久免费精品| 亚洲成人精品av| 亚洲xxx视频| 国产精品流白浆视频| 性欧美xxxx交| 亚洲娇小xxxx欧美娇小| 国产日本欧美一区二区三区| 亚洲新中文字幕| 91高清视频在线免费观看| 成人天堂噜噜噜| 久久久精品国产网站| 国产v综合v亚洲欧美久久| 欧美一区二区三区……| 国产精品入口日韩视频大尺度| 亚洲欧洲成视频免费观看| 精品久久久久久久久中文字幕| 欧美激情一区二区三区在线视频观看| 国产欧美日韩免费| 国产精品激情av电影在线观看| 国产日韩欧美在线视频观看| 国产不卡av在线免费观看| 国产精品视频一区国模私拍| 精品国产一区久久久| 亚洲天堂av高清| 欧美大肥婆大肥bbbbb| 亚洲欧美色婷婷| 久久国产精品视频| 韩国精品美女www爽爽爽视频| 国产一级揄自揄精品视频| 欧美精品日韩三级| 社区色欧美激情 | 国产欧美亚洲精品| 久久精品国产免费观看| 国产午夜精品一区二区三区| 97国产suv精品一区二区62| 日韩中文字在线| 超碰91人人草人人干| 欧美在线性爱视频| 欧美一级在线播放| 欧美成人午夜激情在线| 欧美日韩国产综合视频在线观看中文| 日本精品在线视频| 日韩欧美a级成人黄色| 自拍偷拍亚洲欧美| 国产精品第七影院| 欧美国产日韩二区| 久久久噜噜噜久噜久久| 日韩欧美在线视频| 欧美激情一区二区三区久久久| 庆余年2免费日韩剧观看大牛| 日韩免费在线视频| 国产精品福利小视频| 欧美中文字幕在线| 久久久91精品国产一区不卡| 海角国产乱辈乱精品视频| 国产成人拍精品视频午夜网站| 亚洲欧美国产制服动漫| 狠狠干狠狠久久| 亚洲综合中文字幕68页| 亚洲欧美日本精品| 亚洲人成在线观| 成人性生交大片免费看小说| 欧美日韩国产成人高清视频| 国产一区二区视频在线观看| 亚洲欧美日本精品| 亚洲第一精品自拍| 欧美成人精品一区二区三区| 欧美成人全部免费| 久久久视频免费观看| 欧美视频精品一区| 一区二区三区在线播放欧美| 伦伦影院午夜日韩欧美限制| 日韩欧美一区二区在线| 国产精品丝袜白浆摸在线| 国产精品老牛影院在线观看| 永久免费毛片在线播放不卡| 成人激情综合网| 精品日韩中文字幕| 成人国产亚洲精品a区天堂华泰| 欧美久久久精品| 久久影院免费观看| 国产亚洲欧美aaaa| 欧美激情精品久久久久久蜜臀| 国产色综合天天综合网| 国产日产欧美a一级在线| 2025国产精品视频| 日韩中文字幕在线免费观看| 欧美日韩国产精品| 亚洲欧美国产视频| 久久色在线播放| 国产精品视频免费在线| 韩国精品美女www爽爽爽视频| 亚洲午夜精品久久久久久久久久久久| 久久精品国产91精品亚洲| 欧美精品在线第一页| 色哟哟亚洲精品一区二区| 精品在线欧美视频| 国产精品福利小视频| 国产日韩精品在线| 亚洲欧洲激情在线| 国产成人精品电影久久久| 久久久久国色av免费观看性色| 亚洲人成网站免费播放| 精品视频在线导航| 国产噜噜噜噜久久久久久久久| 91国自产精品中文字幕亚洲| 日韩黄色高清视频| 日韩av手机在线| 最近2019年手机中文字幕| 欧美精品在线免费观看| 亚洲视频在线免费看| 136fldh精品导航福利| 一区二区三区四区精品| 欧美极品在线视频| 97视频免费看| 国模极品一区二区三区| 日韩经典中文字幕在线观看| 国产成人精品a视频一区www| 欧美国产激情18| 日韩精品视频免费| 亚洲成人av片在线观看| 亚洲欧美日本伦理| 国产日产欧美a一级在线| 欧美国产日韩在线| 国产一区二区三区在线视频| 午夜精品视频在线| 中文综合在线观看| 欧美精品videos性欧美| 97av在线视频免费播放| 91久久在线播放| 国产日韩在线看| 欧美性猛交xxxxx免费看| 国产精品入口免费视频一| 欧美性高潮床叫视频|