国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

聽說,99% 的 Go 程序員都被 Defer 坑過

開發 后端
直接把我珍藏多年的代碼一把梭,憑借多年踩坑經歷和寫 BUG 經驗,我要站著把這個坑邁過去。

[[429635]]

先聲明:我被坑過。

之前寫 Go 專欄時,寫過一篇文章:Go 專欄|錯誤處理:defer,panic 和 recover。有小伙伴留言說:道理都懂,但還是不知道怎么用,而且還總出現莫名奇妙的問題。

出問題就對了,這個小東西壞的很,一不留神就出錯。

所以,面對這種情況,我們今天就不講道理了。直接把我珍藏多年的代碼一把梭,憑借多年踩坑經歷和寫 BUG 經驗,我要站著把這個坑邁過去。

一、

先來一個簡單的例子熱熱身:

  1. package main 
  2.  
  3. import ( 
  4.     "fmt" 
  5.  
  6. func main() { 
  7.     defer func() { 
  8.         fmt.Println("first"
  9.     }() 
  10.  
  11.     defer func() { 
  12.         fmt.Println("second"
  13.     }() 
  14.  
  15.     fmt.Println("done"

輸出:

  1. done 
  2. second 
  3. first 

這個比較簡單,defer 語句的執行順序是按調用 defer 語句的倒序執行。

二、

看看這段代碼有什么問題?

  1. for _, filename := range filenames { 
  2.     f, err := os.Open(filename) 
  3.     if err != nil { 
  4.         return err 
  5.     } 
  6.     defer f.Close() 

這段代碼其實很危險,很可能會用盡所有文件描述符。因為 defer 語句不到函數的最后一刻是不會執行的,也就是說文件始終得不到關閉。所以切記,一定不要在 for 循環中使用 defer 語句。

那怎么優化呢?可以將循環體單獨寫一個函數,這樣每次循環的時候都會調用關閉函數。

如下:

  1. for _, filename := range filenames { 
  2.     if err := doFile(filename); err != nil { 
  3.         return err 
  4.     } 
  5.  
  6. func doFile(filename string) error { 
  7.     f, err := os.Open(filename) 
  8.     if err != nil { 
  9.         return err 
  10.     } 
  11.     defer f.Close() 

三、

看看這三個函數的輸出結果是什么?

  1. package main 
  2.  
  3. import ( 
  4.     "fmt" 
  5.  
  6. func a() (r int) { 
  7.     defer func() { 
  8.         r++ 
  9.     }() 
  10.     return 0 
  11.  
  12. func b() (r int) { 
  13.     t := 5 
  14.     defer func() { 
  15.         t = t + 5 
  16.     }() 
  17.     return t 
  18.  
  19. func c() (r int) { 
  20.     defer func(r int) { 
  21.         r = r + 5 
  22.     }(r) 
  23.     return 1 
  24.  
  25. func main() { 
  26.     fmt.Println("a = ", a()) 
  27.     fmt.Println("b = ", b()) 
  28.     fmt.Println("c = ", c()) 

公布答案:

  1. a =  1 
  2. b =  5 
  3. c =  1 

你答對了嗎?

說實話剛開始看到這個結果時,我是相當費解,完全不知道怎么回事。

但可以看到,這三個函數都有一個共同特點,它們都有一個命名返回值,并且都在函數中引用了這個返回值。

引用的方式分兩種:分別是閉包和函數參數。

先看 a() 函數:

閉包通過 r++ 修改了外部變量,返回值變成了 1。

相當于:

  1. func aa() (r int) { 
  2.     r = 0 
  3.     // 在 return 之前,執行 defer 函數 
  4.     func() { 
  5.         r++ 
  6.     }() 
  7.     return 

再看 b() 函數:

閉包內修改的只是局部變量 t,而外部變量 t 不受影響,所以還是返回 5。

相當于:

  1. func bb() (r int) { 
  2.     t := 5 
  3.     // 賦值 
  4.     r = t 
  5.     // 在 return 之前,執行 defer 函數 
  6.     // defer 函數沒有對返回值 r 進行修改,只是修改了變量 t 
  7.     func() { 
  8.         t = t + 5 
  9.     }() 
  10.     return 

最后是 c 函數:

參數傳遞是值拷貝,實參不受影響,所以還是返回 1。

相當于:

  1. func cc() (r int) { 
  2.     // 賦值 
  3.     r = 1 
  4.     // 這里修改的 r 是函數形參的值 
  5.     // 值拷貝,不影響實參值 
  6.     func(r int) { 
  7.         r = r + 5 
  8.     }(r) 
  9.     return 

那么,為了避免寫出這么令人意外的代碼,最好在定義函數時就不要使用命名返回值。或者如果使用了,就不要在 defer 中引用。

再看下面兩個例子:

  1. func d() int { 
  2.     r := 0 
  3.     defer func() { 
  4.         r++ 
  5.     }() 
  6.     return r 
  7.  
  8. func e() int { 
  9.     r := 0 
  10.     defer func(i int) { 
  11.         i++ 
  12.     }(r) 
  13.     return 0 
  14. d =  0 
  15. e =  0 

返回值符合預期,再也不用絞盡腦汁猜了。

四、

defer 表達式的函數如果在 panic 后面,則這個函數無法被執行。

  1. func main() { 
  2.     panic("a"
  3.     defer func() { 
  4.         fmt.Println("b"
  5.     }() 

輸出如下,b 沒有打印出來。

  1. panic: a 
  2.  
  3. goroutine 1 [running]: 
  4. main.main() 
  5.     xxx.go:87 +0x4ce 
  6. exit status 2 

而如果 defer 在前,則可以執行。

  1. func main() { 
  2.     defer func() { 
  3.         fmt.Println("b"
  4.     }() 
  5.     panic("a"

輸出:

  1. panic: a 
  2.  
  3. goroutine 1 [running]: 
  4. main.main() 
  5.     xxx.go:90 +0x4e7 
  6. exit status 2 

五、

看看下面這段代碼的執行順序:

  1. func G() { 
  2.     defer func() { 
  3.         fmt.Println("c"
  4.     }() 
  5.  
  6.     F() 
  7.     fmt.Println("繼續執行"
  8.  
  9. func F() { 
  10.     defer func() { 
  11.         if err := recover(); err != nil { 
  12.             fmt.Println("捕獲異常:", err) 
  13.         } 
  14.         fmt.Println("b"
  15.     }() 
  16.     panic("a"
  17.  
  18. func main() { 
  19.     G() 

順序如下:

  1. 調用 G() 函數;
  2. 調用 F() 函數;
  3. F() 中遇到 panic,立刻終止,不執行 panic 之后的代碼;
  4. 執行 F() 中 defer 函數,遇到 recover 捕獲錯誤,繼續執行 defer 中代碼,然后返回;
  5. 執行 G() 函數后續代碼,最后執行 G() 中 defer 函數。

輸出:

  1. 捕獲異常: a 
  2. 繼續執行 

五、

看看下面這段代碼的執行順序:

  1. func G() { 
  2.     defer func() { 
  3.         if err := recover(); err != nil { 
  4.             fmt.Println("捕獲異常:", err) 
  5.         } 
  6.         fmt.Println("c"
  7.     }() 
  8.  
  9.     F() 
  10.     fmt.Println("繼續執行"
  11.  
  12. func F() { 
  13.     defer func() { 
  14.         fmt.Println("b"
  15.     }() 
  16.     panic("a"
  17.  
  18. func main() { 
  19.     G() 

順序如下:

  1. 調用 G() 函數;
  2. 調用 F() 函數;
  3. F() 中遇到 panic,立刻終止,不執行 panic 之后的代碼;
  4. 執行 F() 中 defer 函數,由于沒有 recover,則將 panic 拋到 G() 中;
  5. G() 收到 panic 則不會執行后續代碼,直接執行 defer 函數;
  6. defer 中捕獲 F() 拋出的異常 a,然后繼續執行,最后退出。

輸出:

  1. 捕獲異常: a 

六、

看看下面這段代碼的執行順序:

  1. func G() { 
  2.     defer func() { 
  3.         fmt.Println("c"
  4.     }() 
  5.  
  6.     F() 
  7.     fmt.Println("繼續執行"
  8.  
  9. func F() { 
  10.     defer func() { 
  11.         fmt.Println("b"
  12.     }() 
  13.     panic("a"
  14.  
  15. func main() { 
  16.     G() 

順序如下:

  1. 調用 G() 函數;
  2. 調用 F() 函數;
  3. F() 中遇到 panic,立刻終止,不執行 panic 之后的代碼;
  4. 執行 F() 中 defer 函數,由于沒有 recover,則將 panic 拋到 G() 中;
  5. G() 收到 panic 則不會執行后續代碼,直接執行 defer 函數;
  6. 由于沒有 recover,直接拋出 F() 拋過來的異常 a,然后退出。

輸出:

  1. panic: a 
  2.  
  3. goroutine 1 [running]: 
  4. main.F() 
  5.     xxx.go:90 +0x5b 
  6. main.G() 
  7.     xxx.go:82 +0x48 
  8. main.main() 
  9.     xxx.go:107 +0x4a5 
  10. exit status 2 

七、

看看下面這段代碼的執行順序:

  1. func G() { 
  2.     defer func() { 
  3.         // goroutine 外進行 recover 
  4.         if err := recover(); err != nil { 
  5.             fmt.Println("捕獲異常:", err) 
  6.         } 
  7.         fmt.Println("c"
  8.     }() 
  9.  
  10.     // 創建 goroutine 調用 F 函數 
  11.     go F() 
  12.     time.Sleep(time.Second
  13.  
  14. func F() { 
  15.     defer func() { 
  16.         fmt.Println("b"
  17.     }() 
  18.     // goroutine 內部拋出panic 
  19.     panic("a"
  20.  
  21. func main() { 
  22.     G() 

順序如下:

  1. 調用 G() 函數;
  2. 通過 goroutine 調用 F() 函數;
  3. F() 中遇到 panic,立刻終止,不執行 panic 之后的代碼;
  4. 執行 F() 中 defer 函數,由于沒有 recover,則將 panic 拋到 G() 中;
  5. 由于 goroutine 內部沒有進行 recover,則 goroutine 外部函數,也就是 G() 函數是沒辦法捕獲的,程序直接崩潰退出。

輸出:

  1. panic: a 
  2.  
  3. goroutine 6 [running]: 
  4. main.F() 
  5.     xxx.go:96 +0x5b 
  6. created by main.G 
  7.     xxx.go:87 +0x57 
  8. exit status 2 

八、

最后再說一個 recover 的返回值問題:

  1. defer func() { 
  2.     if err := recover(); err != nil { 
  3.         fmt.Println("捕獲異常:", err.Error()) 
  4.     } 
  5. }() 
  6. panic("a"

recover 返回的是 interface {} 類型,而不是 error 類型,所以這樣使用的話會報錯:

  1. err.Error undefined (type interface {} is interface with no methods) 

可以這樣來轉換一下:

  1. defer func() { 
  2.     if err := recover(); err != nil { 
  3.         fmt.Println("捕獲異常:", fmt.Errorf("%v", err).Error()) 
  4.     } 
  5. }() 
  6. panic("a"

或者直接打印結果:

  1. defer func() { 
  2.     if err := recover(); err != nil { 
  3.         fmt.Println("捕獲異常:", err) 
  4.     } 
  5. }() 
  6. panic("a"

輸出:

  1. 捕獲異常: a 

以上就是本文的全部內容,其實寫過其他的語言的同學都知道,關閉文件句柄,釋放鎖等操作是很容易忘的。而 Go 語言通過 defer 很好地解決了這個問題,但在使用過程中還是要小心。

本文總結了一些容踩坑的點,希望能夠幫助大家少寫 BUG,如果大家覺得有用的話,歡迎點贊和轉發。

文章中的腦圖和源碼都上傳到了 GitHub,有需要的同學可自行下載。

源碼地址:

https://github.com/yongxinz/gopher/tree/main/sc

本文轉載自微信公眾號「AlwaysBeta」,可以通過以下二維碼關注。轉載本文請聯系AlwaysBeta公眾號。

 

責任編輯:武曉燕 來源: AlwaysBeta
相關推薦

2025-04-03 12:30:00

C 語言隱式類型轉換代碼

2022-07-15 08:20:54

Java基礎知識

2020-10-09 07:54:43

PythonJava爬蟲

2025-04-29 08:30:00

迭代器失效C++編程

2019-10-25 22:17:25

開發者技能工具

2025-04-21 10:35:37

2020-08-05 07:53:53

程序員網站技術

2018-02-06 08:36:02

簡歷程序員面試

2015-05-15 10:09:09

程序員

2024-03-26 00:48:38

2020-09-14 08:47:46

緩存程序員存儲

2015-09-16 09:57:41

swoolePHP程序員

2024-04-01 08:05:27

Go開發Java

2023-11-13 08:34:01

Java編程習慣

2024-03-13 13:10:48

JavaInteger緩存

2014-08-13 11:11:58

程序員

2020-10-28 09:43:40

前端開發Vue

2022-09-25 21:58:27

程序員

2013-08-20 09:33:59

程序員

2018-10-11 10:41:12

Go 開發技術
點贊
收藏

51CTO技術棧公眾號

色中色一区二区| 亚洲国产精品欧美一二99| 美女黄色丝袜一区| 九九在线视频| 久久久国产精品午夜一区ai换脸| 精品视频在线观看| 欧美肉体xxxx裸体137大胆| xxxxx成人.com| 国内精品久久久久久野外| 亚洲一区电影777| www.色就是色| 99在线精品视频| 欧美日韩综合另类| 欧美xxx在线观看| 成人a在线视频| 成人中文在线| 国产精品久久久久久久久久免费| 中文字幕亚洲在线观看| 在线观看精品自拍私拍| 一级毛片久久久| 精品国产人成亚洲区| 国产在线高潮| 日韩欧美成人午夜| 欧美理论片在线播放| 欧美一区二区美女| www免费在线观看| 91精品国产福利| 激情视频在线观看| 欧美麻豆精品久久久久久| av播放在线观看| 在线综合+亚洲+欧美中文字幕| 国产高清视频在线| 欧美猛男超大videosgay| 免费高清完整在线观看| 日韩视频免费观看高清完整版 | 欧美精品黄色| 成人h视频在线观看| 国产亚洲综合精品| 中文精品一区二区三区| 大白屁股一区二区视频| 日韩大片一区二区| 亚洲国产三级在线| 日本在线免费| 亚洲欧美国产一区二区三区| 精品一区二区三区免费看| 国外成人在线播放| 欧美日本一区| 超级碰在线观看| 亚洲天堂成人在线观看| 午夜激情在线观看| 在线电影av不卡网址| 亚洲理论电影片| 亚洲精品日韩av| 美女免费视频一区| 欧美女人性生活视频| 一区二区三区美女视频| 男人的天堂在线视频免费观看 | 韩国中文字幕在线| 亚洲精品在线看| 日本高清久久| 91欧美激情另类亚洲| 日本视频免费一区| 天天色综合天天色| 精品1区2区3区| 欧美成人h版| 奇米一区二区三区四区久久| 亚洲私拍自拍| 日韩精品 欧美| 电影天堂久久| 日本高清成人免费播放| 欧美裸体视频| 日本精品视频在线观看| 媚黑女一区二区| 国产野外作爱视频播放| 在线观看91av| 成人在线超碰| 女同一区二区| 国产精品久久久久久久久晋中 | 成人在线小说| 亚洲国产精品一区二区久久| 国产网站在线免费观看| 久久天天躁狠狠躁夜夜躁| 成人高清电影网站| 亚洲国产日韩欧美| 18欧美乱大交hd1984| 青青青国内视频在线观看软件| 九九热这里只有精品6| 伊人成年综合电影网| 国产精品秘入口18禁麻豆免会员| 亚洲成在人线免费| 91精品国产66| 国产日韩欧美二区| 中文乱码免费一区二区| 成人黄色在线电影| 欧美一级成年大片在线观看| 日本91福利区| 手机亚洲第一页| 欧美激情xxxxx| 久久91精品久久久久久秒播| 男男激情在线| 热久久这里只有精品| 白白色亚洲国产精品| 久草资源在线| 国产日韩在线看| 欧美激情综合在线| 免费成人在线电影| 99在线高清视频在线播放| 国产精品午夜电影| 日韩电影大全网站| 日本免费高清一区| 色综合久久久久久久久久久| 免费看成人人体视频| 日本网站免费在线观看| 亚洲成人av中文字幕| 亚洲日本欧美| 四虎精品在线| 国产精品亚洲视频在线观看| 日本一区二区动态图| 久久久久黄色| 久久福利一区二区| 亚洲国产一区自拍| 快she精品国产999| 加勒比一区二区三区在线| 性亚洲最疯狂xxxx高清| 91亚洲国产成人精品一区二区三| av小说在线播放| 日本在线观看一区| 欧美日韩国产高清一区二区三区| 欧美特黄一级大片| 8×8x拔擦拔擦在线视频网站| 亚洲 日韩 国产第一| 95精品视频在线| 国产一区二区三区四区五区3d| 免费成人深夜夜行网站视频| 亚洲二区中文字幕| 葵司免费一区二区三区四区五区| 黄视频在线观看免费| 成人在线视频网| 欧美性生交xxxxx久久久| 国产精品国产三级国产在线观看 | 国产成人综合在线观看| 青青草超碰在线| 国产日韩精品在线| 香蕉久久一区二区不卡无毒影院| 麻豆一区一区三区四区| 69国产精品视频| 国产精品电影网| 欧美日韩日本国产| 欧美视频导航| 国产精品久久久久久福利| 日韩高清专区| 亚洲性夜色噜噜噜7777| 丁香激情综合五月| 91精品在线免费视频| 欧美日韩在线成人| 欧美有码在线观看视频| 亚洲午夜av在线| 亚洲人成免费| 国产高清视频色在线www| 欧美日韩黄色一级片| 性欧美xxxx视频在线观看| 亚洲综合一区在线| 欧美日韩1区2区3区| 欧美理论片在线播放| 成 年 人 黄 色 大 片大 全| 美女少妇精品视频| 亚洲一区二区三区美女| 伊人久久久大香线蕉综合直播| www在线看| 在线观看免费黄网站| 国产精品久久久一区| 欧美日韩成人综合天天影院| 蜜桃一区二区三区在线观看| 成人在线免费| 69国产精品| 日本一区二区三区免费看| 中文字幕在线日韩| 亚洲精品一二三区| 男人天堂欧美日韩| 精品伊人久久| 激情小视频在线| 好吊色视频988gao在线观看| 欧美在线视频a| 精品久久久久久最新网址| 欧美激情在线看| 激情欧美亚洲| 激情五月综合婷婷| 女人天堂在线| youjizz.com在线观看| 国产精品免费久久久久影院| 精品国产凹凸成av人网站| 中文字幕亚洲在| 男女男精品视频| 精品产国自在拍| videos性欧美另类高清| **毛片在线网站| 性高湖久久久久久久久aaaaa| 51国偷自产一区二区三区| 日韩中文字幕在线视频| 欧美巨大另类极品videosbest|