GO常见的错误99%程序员会遇到

新学习go语言的人可能遇到常见的错误,其中有两个比较常见的错误,需要单独拿出来说下,为什么要单独说呢,因为这两个错误跟其他语言不同,是因为go本身的设计造成的。

 

在循环(迭代)中使用了变量的引用

在go语言中,循环(迭代)所使用的变量是同一个变量,只是在每次循环的时候被赋于不同的值,这样的做的目的呢,当然是出于高效考虑咯。但是,如果使用不当的话,可能会引起意想不到的行为。

举一个栗子:

func main() {
	var out []*int
	for i := 0; i < 3; i++ {
		out = append(out, &i)
	}
	fmt.Println("Values:", *out[0], *out[1], *out[2])
	fmt.Println("Addresses:", out[0], out[1], out[2])
}

上面的代码会输出:

Values: 3 3 3
Addresses: 0x40e020 0x40e020 0x40e020

因为每次循环中,我们只是把变量 i 的地址放进 out 数组里,因为变量 i 是同一个变量,只有在循环结束的时候,被赋值为3。

解决方法:申明一个新的变量

for i := 0; i < 3; i++ {
	i := i // Copy i into a new variable.
 	out = append(out, &i)
 }

结果

Values: 0 1 2
Addresses: 0x40e020 0x40e024 0x40e028

同理对于切片来说,也用有这个问题,因为切片本身就只是一个地址而已

func main() {
	var out [][]int
	for _, i := range [][1]int{{1}, {2}, {3}} {
		out = append(out, i[:])
	}
	fmt.Println("Values:", out)
}

结果:

Values: [[3] [3] [3]]

同样的问题,在循环里使用协程也会遇到

vi设计http://www.maiqicn.com 办公资源网站大全https://www.wode007.com

在协程中使用循环变量

按照程序员的思维,都喜欢使用并发,你可能会写出下面的代码: 心里特别开心,原来go 的并发这么简单。

for _, val := range values {
	go func() {
    	fmt.Println(val)
	}()
}

但是,你可能会发现输出的结果是一摸一样的! 因为go的协程跑起来也是需要一点时间的,循环结束的时候,可能一个goroute都没有跑完,然后 val 值确被赋值了,所以,你会看到,输出的都是最后一个值

解决方法:

for _, val := range values {
	go func(val interface{}) {
		fmt.Println(val)
	}(val)
}

当然也可以

for i := range valslice {
	val := valslice[i]
	go func() {
		fmt.Println(val)
	}()
}
posted @ 2020-10-04 13:37  浅笑·  阅读(154)  评论(0编辑  收藏  举报