成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Go 有哪幾種無法恢復(fù)的致命場景?

開發(fā) 后端
在今天這篇文章中,我們介紹了 Go 語言的三種錯誤類型。其中針對大家最少見,但一碰到就很容易翻車的致命錯誤 fatal error 進行了介紹,給出了一些經(jīng)典案例。

[[442134]]

 大家好,我是煎魚。

有一次事故現(xiàn)場,在緊急恢復(fù)后,他正在排查代碼,查了好一會。我回頭一看,這錯誤提醒很明顯就是致命錯誤,較好定位。

但此時,他竟然在查 panic-recover 是不是哪里漏了,我表示大受震驚...

今天就由煎魚給大家分享一下錯誤類型有哪幾種,又在什么場景下會觸發(fā)。

錯誤類型

error

第一種是 Go 中最標準的 error 錯誤,其真身是一個 interface{}。

如下:

  1. type error interface { 
  2.     Error() string 

在日常工程中,我們只需要創(chuàng)建任意結(jié)構(gòu)體,實現(xiàn)了 Error 方法,就可以認為是 error 錯誤類型。

如下:

  1. type errorString struct { 
  2.     s string 
  3.  
  4. func (e *errorString) Error() string { 
  5.     return e.s 

在外部調(diào)用標準庫 API,一般如下:

  1. f, err := os.Open("filename.ext"
  2. if err != nil { 
  3.     log.Fatal(err) 
  4. // do something with the open *File f 

我們會約定最后一個參數(shù)為 error 類型,一般常見于第二個參數(shù),可以有個約定俗成的習慣。

panic

第二種是 Go 中的異常處理 panic,能夠產(chǎn)生異常錯誤,結(jié)合 panic+recover 可以扭轉(zhuǎn)程序的運行狀態(tài)。

如下:

  1. package main 
  2.  
  3. import "os" 
  4.  
  5. func main() { 
  6.     panic("a problem"
  7.  
  8.     _, err := os.Create("/tmp/file"
  9.     if err != nil { 
  10.         panic(err) 
  11.     } 

輸出結(jié)果:

  1. $ go run panic.go 
  2. panic: a problem 
  3. goroutine 1 [running]: 
  4. main.main() 
  5.     /.../panic.go:12 +0x47 
  6. ... 
  7. exit status 2 

如果沒有使用 recover 作為捕獲,就會導(dǎo)致程序中斷。也因此經(jīng)常被人誤以為程序中斷,就 100% 是 panic 導(dǎo)致的。

這是一個誤區(qū)。

throw

第三種是 Go 初學者經(jīng)常踩坑,也不知道的錯誤類型,那就是致命錯誤 throw。

這個錯誤類型,在用戶側(cè)是沒法主動調(diào)用的,均為 Go 底層自行調(diào)用的,像是大家常見的 map 并發(fā)讀寫,就是由此觸發(fā)。

其源碼如下:

  1. func throw(s string) { 
  2.  systemstack(func() { 
  3.   print("fatal error: ", s, "\n"
  4.  }) 
  5.  gp := getg() 
  6.  if gp.m.throwing == 0 { 
  7.   gp.m.throwing = 1 
  8.  } 
  9.  fatalthrow() 
  10.  *(*int)(nil) = 0 // not reached 

根據(jù)上述程序,會獲取當前 G 的實例,并設(shè)置其 M 的 throwing 狀態(tài)為 1。

狀態(tài)設(shè)置好后,會調(diào)用 fatalthrow 方法進行真正的 crash 相關(guān)操作:

  1. func fatalthrow() { 
  2.  pc := getcallerpc() 
  3.  sp := getcallersp() 
  4.  gp := getg() 
  5.   
  6.  systemstack(func() { 
  7.   startpanic_m() 
  8.   if dopanic_m(gp, pc, sp) { 
  9.    crash() 
  10.   } 
  11.  
  12.   exit(2) 
  13.  }) 
  14.  
  15.  *(*int)(nil) = 0 // not reached 

主體邏輯是發(fā)送 _SIGABRT 信號量,最后調(diào)用 exit 方法退出,所以你會發(fā)現(xiàn)這是攔也攔不住的 “致命” 錯誤。

致命場景

為此,作為一名 “成熟” 的 Go 工程師,除了保障自己程序的健壯性外,我也在網(wǎng)上收集了一些致命的錯誤場景,分享給大家。

一起學習和規(guī)避這些致命場景,年底爭取拿個 A,不要背上 P0 事故。

并發(fā)讀寫 map

  1. func foo() { 
  2.  m := map[string]int{} 
  3.  go func() { 
  4.   for { 
  5.    m["煎魚1"] = 1 
  6.   } 
  7.  }() 
  8.  for { 
  9.   _ = m["煎魚2"
  10.  } 

輸出結(jié)果:

  1. fatal error: concurrent map read and map write 
  2.  
  3. goroutine 1 [running]: 
  4. runtime.throw(0x1078103, 0x21) 
  5. ... 

堆棧內(nèi)存耗盡

  1. func foo() { 
  2.  var f func(a [1000]int64) 
  3.  f = func(a [1000]int64) { 
  4.   f(a) 
  5.  } 
  6.  f([1000]int64{}) 

輸出結(jié)果:

  1. runtime: goroutine stack exceeds 1000000000-byte limit 
  2. runtime: sp=0xc0200e1bf0 stack=[0xc0200e0000, 0xc0400e0000] 
  3. fatal error: stack overflow 
  4.  
  5. runtime stack: 
  6. runtime.throw(0x1074ba3, 0xe) 
  7.         /usr/local/Cellar/go/1.16.6/libexec/src/runtime/panic.go:1117 +0x72 
  8. runtime.newstack() 
  9. ... 

將 nil 函數(shù)作為 goroutine 啟動

  1. func foo() { 
  2.  var f func() 
  3.  go f() 

輸出結(jié)果:

  1. fatal error: go of nil func value 
  2.  
  3. goroutine 1 [running]: 
  4. main.foo() 
  5. ... 

goroutines 死鎖

  1. func foo() { 
  2.  select {} 

輸出結(jié)果:

  1. fatal error: all goroutines are asleep - deadlock! 
  2.  
  3. goroutine 1 [select (no cases)]: 
  4. main.foo() 
  5. ... 

線程限制耗盡

如果你的 goroutines 被 IO 操作阻塞了,新的線程可能會被啟動來執(zhí)行你的其他 goroutines。

Go 的最大的線程數(shù)是有默認限制的,如果達到了這個限制,你的應(yīng)用程序就會崩潰。

會出現(xiàn)如下輸出結(jié)果:

  1. fatal error: thread exhaustion 
  2. ... 

可以通過調(diào)用 runtime.SetMaxThreads 方法增大線程數(shù),不過也需要考量是否程序有問題。

超出可用內(nèi)存

如果你執(zhí)行的操作,例如:下載大文件等。導(dǎo)致應(yīng)用程序占用內(nèi)存過大,程序上漲,導(dǎo)致 OOM。

會出現(xiàn)如下輸出結(jié)果:

  1. fatal error: runtime: out of memory 
  2. ... 

建議處理掉一些程序,或者換新電腦了。

總結(jié)

在今天這篇文章中,我們介紹了 Go 語言的三種錯誤類型。其中針對大家最少見,但一碰到就很容易翻車的致命錯誤 fatal error 進行了介紹,給出了一些經(jīng)典案例。

希望大家后續(xù)能夠規(guī)避,你有沒有遇到過其中的場景?

歡迎在評論區(qū)交流和留言:)

參考

Are all runtime errors recoverable in Go?

 

責任編輯:武曉燕 來源: 腦子進煎魚了
相關(guān)推薦

2011-09-01 09:39:06

2021-12-20 23:24:40

前端測試開發(fā)

2018-07-28 00:20:15

2024-05-27 09:07:27

2022-04-29 13:40:55

前端測試后端

2010-08-20 10:26:25

DB2數(shù)據(jù)類型

2024-10-09 09:12:11

2019-06-06 15:48:21

筆記本無線網(wǎng)卡

2024-04-15 10:30:22

MySQL存儲引擎

2021-07-14 08:00:13

reactCss模塊

2024-05-10 08:34:26

Python內(nèi)存對象

2025-06-12 09:30:25

2020-07-11 09:42:59

python數(shù)據(jù)挖掘數(shù)據(jù)分析

2010-08-17 13:00:19

DB2數(shù)據(jù)遷移

2019-04-30 10:00:59

CSS居中前端

2011-05-26 10:01:16

Oracle數(shù)據(jù)庫ODU

2010-08-16 10:53:33

DB2 9管理軟件安裝

2025-03-31 07:53:10

單例模式設(shè)計模式C#

2024-05-15 16:54:04

2024-01-09 11:56:58

Go編程語言
點贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 日日摸天天添天天添破 | 精品一区二区在线观看 | 国产精品电影在线观看 | 成年人免费看的视频 | aaa精品| 亚洲精选久久 | 精品一级电影 | 亚洲一区二区三区四区五区中文 | 日韩视频专区 | 大陆一级毛片免费视频观看 | 色视频在线播放 | 97伊人 | 日韩精品一区二区三区中文在线 | 91大神在线资源观看无广告 | 国产日韩欧美综合 | 欧美天堂| 大象一区 | 一区二区三区欧美在线 | 蜜桃日韩| 亚洲精品在线播放 | 成人欧美一区二区三区在线观看 | 欧美福利视频 | 欧美bondage紧缚视频 | 精品久久久久久久久久久 | 黄色av网站免费看 | 一级久久久久久 | 天天射影院 | 成人午夜看片 | 天天操天天干天天透 | 免费在线观看黄视频 | 精品一区二区三区在线观看 | 一区二区视频在线 | 911网站大全在线观看 | 久久国产亚洲 | 欧美成年视频 | 97成人免费| 91精品国产91久久久久久密臀 | 91精品久久久 | 色欧美片视频在线观看 | 日本高清视频在线播放 | 日韩欧美网 |