[golang 易犯错误] golang 局部变量初始化:=的陷阱

    我们知道,golang中局部变量初始化方法(使用“:=”创建并赋值),让我们在使用变量时很方便。但是,这也是易犯错误的地方之一。特别是这个初始化符还支持多个变量同时初始化,更特别的是它还支持原有变量赋值和新变量创建并赋值同时进行!也就是说如果有部分变量不存在的而另外一些是已声明好的,用:=来初始化部分变量同样有效。这其实也没什么,更方便了嘛。但是,go的好多语句还支持局部前置语句,比如在if,for,switch等语句的初始化条件语句中。在这些地方,当你以为使用了原有变量的时候,实际上go已经为你创建了一个局限于这些语句作用范围的新变量!

    文字功力太低,说了半天你可能更糊涂了,还是直接用代码说明吧。

//这段代码,一开始声明了code,然后用Atoi解析,此时err没有声明,所以使用了局部变量创建赋值“:=”
//下面的写法正确,code还是已存在的那个变量,正确解析后已经变为5
func test(){ var code int
scode:="5" code,err:=strconv.Atoi(scode); if err!=nil{ return err; } return code; }

//如果,将上面的函数改为下面的样子,结果就不对了,因为在if的后面的初始化语句中使用:=实际上告诉go创建新的在if语句作用域下的局部变量!
//此时,code和err都是新的变量,因此,解析后语句外部的code不受影响,结果错误!
func test(){
var code int
scode:="5"
if code,err:=strconv.Atoi(scode); err!=nil{
return err;
}
return code;
}

下面,用一段完整的代码演示一下。

package main

import "fmt"
import "strconv"

func main() {
    fmt.Println("======conv func 1:")
    conv1()
    fmt.Println("======conv func 2:")
    conv2()
}
func conv1(){
    code:=10
    scode:="5"
    code,err :=strconv.Atoi(scode);
    if err!=nil{
        fmt.Println("convert error",err," code=",code);
    }
    fmt.Println("after conv ,code=",code);
}
func conv2(){
    code:=10
    scode:="5"
    if code,err :=strconv.Atoi(scode);err!=nil{
        fmt.Println("convert error",err," code=",code);
    }
    fmt.Println("after conv ,code=",code);
}

代码的执行结果:

======conv func 1:
after conv ,code= 5
======conv func 2:
after conv ,code= 10

Program exited.

 

posted @ 2015-01-08 20:36  柒零壹  阅读(3510)  评论(0编辑  收藏  举报