這是一個建立於 的文章,其中的資訊可能已經有所發展或是發生改變。
最近遇到了一個變數範圍的問題,一個比較低級的問題,可能作為一個熟手不應該犯這樣的低級錯誤,但是golang的文法特點可能讓你稍微不注意就踩坑,嘿嘿。
變數範圍
全域變數的範圍是整個包,局部變數的範圍是該變數所在的花括弧內,這是一個很基礎的問題。我們通常會使用golang的一個文法糖:=
來給變數賦值,這種方式可以節省掉我們定義變數的代碼,讓代碼變的更加簡潔,但是如果你定義了一個全域變數,又不小心用:=
來給它賦值,就會出現一些問題。
問題
看下面的代碼,定義了一個全域變數t
,我想在init()
中給他賦值為2,然後在main中使用它。
var t intfunc init() { t, err := strconv.Atoi("2") if err != nil { log.Fatalln(err) } fmt.Println("init:", t)}func main() { fmt.Println("main:", t)}
輸出:
init: 2main: 0
執行之後,在init和main中列印出了不一樣的數字,為什麼會不一樣呢,可能你仔細一看就知道原因了。很簡單,init中的t是用:=
產生的,所以t是局部變數,在init函數中覆蓋了全域變數t。全域變數t並沒有被賦值,它還是原來的0值。
我本想在init中給全域變數t賦值的,卻不小心用:=
建立了一個局部變數導致全域變數t沒有賦值成功,犯了一個低級錯誤。
解決
知道原因之後就容易解決了,我不使用:=
就可以了。代碼如下:
var t intfunc init() { var err error t, err = strconv.Atoi("2") if err != nil { log.Fatalln(err) } fmt.Println("init:", t)}func main() { fmt.Println("main:", t)}
輸出:
init: 2main: 2
沒有使用:=
之後,init中的t就是全域變數t,給全域變數t賦值為2,main中自然輸出的就是2,實現了我最初的目的。
思考
這個問題很簡單很低級,但是可能一個golang熟手,在代碼遠比demo要複雜的多的實際項目中,不經意間就可能會犯下這樣的錯誤。
這個問題很難保證說下次一定不會出現的,除非你徹底不用全域變數或者徹底不用:=
這樣的文法糖。我的建議是這樣:
- 盡量少的使用全域變數。
- 盡量少的使用
:=
文法糖。
- 使用
:=
的時候要確保左值沒有被定義過。