App下載

Go開發(fā)中的5個常見錯誤分享!

著名奶茶鑒定家 2021-08-26 15:59:21 瀏覽數 (2531)
反饋

錯誤風險是代碼中可能導致生產錯誤和中斷的問題。錯誤是代碼中的缺陷,它會產生不希望的或不正確的結果。由于糟糕的編碼實踐、缺乏版本控制、需求傳達錯誤、不切實際的開發(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。


0 人點贊