錯誤風險是代碼中可能導致生產錯誤和中斷的問題。錯誤是代碼中的缺陷,它會產生不希望的或不正確的結果。由于糟糕的編碼實踐、缺乏版本控制、需求傳達錯誤、不切實際的開發(fā)時間表以及有缺陷的第三方工具,代碼通常存在錯誤風險。在這篇文章中,讓我們來看看 Go 中一些常見的錯誤風險。
1.無限遞歸調用
遞歸調用自身的函數需要有一個退出條件。否則,它將永遠遞歸,直到系統(tǒng)內存耗盡。
此問題可能是由常見錯誤引起的,例如忘記添加退出條件。它也可能“故意”發(fā)生。某些語言具有尾調用優(yōu)化,這使得某些無限遞歸調用可以安全使用。尾調用優(yōu)化允許您避免為函數分配新的堆棧幀,因為調用函數將返回它從被調用函數獲取的值。最常見的用途是尾遞歸,其中為利用尾調用優(yōu)化而編寫的遞歸函數可以使用常量堆棧空間。然而,Go 并沒有實現尾調用優(yōu)化,你最終會耗盡內存。然而,這個問題不適用于產生新的 goroutine。
2. 分配給nil地圖
在添加任何元素之前,需要使用make函數(或map文字)初始化映射。使用內置函數創(chuàng)建一個新的空映射值make,該函數將map類型和可選的容量提示作為參數:
make(map[string]int)
make(map[string]int, 100)
初始容量不限制其大小:地圖增長以容納存儲在其中的項目數量,nil
地圖除外。甲nil
地圖相當于不同之處在于可以添加沒有元素的空映射。
不好的模式:
var countedData map[string][]ChartElement
好的模式:
countedData := make(map[string][]ChartElement)
推薦閱讀:Go:賦值到 nil 映射中的條目
3.方法修改接收器
修改非指針接收器值的方法可能會產生不良后果。這是一個錯誤風險,因為該方法可能會更改方法內部接收器的值,但不會反映在原始值中。要傳播更改,接收者必須是一個指針。
例如:
type data struct {
num int
key *string
items map[string]bool
}
func (d data) vmethod() {
d.num = 8
}
func (d data) run() {
d.vmethod()
fmt.Printf("%+v", d) // Output: {num:1 key:0xc0000961e0 items:map[1:true]}
}
如果num
必須修改:
type data struct {
num int
key *string
items map[string]bool
}
func (d *data) vmethod() {
d.num = 8
}
func (d *data) run() {
d.vmethod()
fmt.Printf("%+v", d) // Output: &{num:8 key:0xc00010a040 items:map[1:true]}
}
4. Goroutine 中可能使用了不需要的值
循環(huán)中的范圍變量在每次迭代中都被重用;因此,在循環(huán)中創(chuàng)建的 goroutine 將指向上作用域的范圍變量。這樣,goroutine 就可以使用帶有不需要的值的變量。
在下面的示例中,goroutine 中使用的 index 和 value 的值來自外部范圍。因為 goroutine 是異步運行的,所以 index 和 value 的值可能(通常是)與預期值不同。
mySlice := []string{"A", "B", "C"}
for index, value := range mySlice {
go func() {
fmt.Printf("Index: %d\n", index)
fmt.Printf("Value: %s\n", value)
}()
}
為了克服這個問題,必須創(chuàng)建一個本地作用域,如下例所示。
mySlice := []string{"A", "B", "C"}
for index, value := range mySlice {
index := index
value := value
go func() {
fmt.Printf("Index: %d\n", index)
fmt.Printf("Value: %s\n", value)
}()
}
處理此問題的另一種方法是將值作為 args 傳遞給 goroutine。
mySlice := []string{"A", "B", "C"}
for index, value := range mySlice {
go func(index int, value string) {
fmt.Printf("Index: %d\n", index)
fmt.Printf("Value: %s\n", value)
}(index, value)
}
推薦閱讀:作為 goroutine 運行的閉包會發(fā)生什么?
5.Close在檢查可能的錯誤之前推遲
對于實現接口的值defer的Close()方法,這是 Go 開發(fā)人員的常見模式io.Closer。例如,打開文件時:
f, err := os.Open("/tmp/file.md")
if err != nil {
return err
}
defer f.Close()
但是這種模式對于可寫文件是有害的,因為推遲函數調用會忽略其返回值,并且該Close()方法可能會返回錯誤。例如,如果您將數據寫入文件,則在您調用Close. 應明確處理此錯誤。
雖然您可以在不使用的情況下繼續(xù),但defer您需要記住每次完成工作時關閉文件。更好的方法是defer使用包裝函數,如下例所示。
f, err := os.Open("/tmp/file.md")
if err != nil {
return err
}
defer func() {
closeErr := f.Close()
if closeErr != nil {
if err == nil {
err = closeErr
} else {
log.Println("Error occured while closing the file :", closeErr)
}
}
}()
return err
推薦閱讀:不要在可寫文件上延遲 Close()
在團隊中工作時,審查其他人的代碼變得很重要。DeepSource是一種自動化代碼審查工具,可管理端到端代碼掃描過程,并在推送新提交或新拉取請求時自動發(fā)出帶有修復的拉取請求。
為 Go 設置 DeepSource非常簡單。一旦設置完成,將對整個代碼庫執(zhí)行初始掃描,找到改進的范圍,修復它們,并為這些更改打開 PR。