1 7 Range深度解析

1.7 range深度解析 #

本节源码位置 https://github.com/golang-minibear2333/golang/blob/master/1.base/1.7-range-deep

1.7.1 range(范围) #

range 关键字在 go 语言中是相当常用好用的语法糖,可以用在 for 循环中迭代 arrayslicemapchannel字符串所有涉及到遍历输出的东西。

1.7.2 怎么用? #

我们在前一节 循环 中初次触及到了 range,也知道他可以省略key,或者省略value来循环遍历的特性,但是这种特性要结合实际情况考量该用哪一个。

切片迭代

	nums := []int{1, 2, 3}
	for k, v := range nums {
		fmt.Printf("key: %v , value: %v  \n", k, v)
	}

这和迭代方式非常适合用for-range语句,如果减少赋值,直接索引num[key]可以减少损耗。如下

for k, _:= range nums {

map迭代 注意,从 Go1开始,遍历的起始节点就是随机了。

	kvs := map[string]string{
		"a":"a",
		"b":"b",
	}
	for k, v := range kvs {
		fmt.Printf("key: %v , value: %v  \n", k, v)
	}

函数中for-range语句中只获取 key 值,然后跟据 key 值获取 value 值,虽然看似减少了一次赋值,但通过 key 值查找 value 值的性能消耗可能高于赋值消耗。

所以能否优化取决于 map 所存储数据结构特征、结合实际情况进行。

字符串迭代(一个一个的输出字符)

for k,v := range "hello"{
  //注意这里单个字符输出的是ASCII码,
  //用 %c 代表输出字符
		fmt.Printf("key: %v , value: %c \n", k, v)
	}

channel (如果不会可以先 mark 下,详细参考后续:go 的并发特性章节)

ch := make(chan int, 10)
	ch <- 11
	ch <- 12

	close(ch) // 不用的时候记得关掉,不关掉又没有另一个goroutine存在会死锁哦,可以注释掉这一句体验死锁

	for x := range ch {
		fmt.Println(x)
	}

结构体

tmp := []struct{
		int
		string
	}{
		{1, "a"},
		{2, "b"},
	}

	for k,v := range tmp{
		fmt.Printf("k:%v, v:%v  \n",k,v)
	}

注意:由于循环开始前循环次数就已经确定了,所以循环过程中新添加的元素是没办法遍历到的。

1.7.3 有可能会遇到的坑! #

由于range遍历时value是值的拷贝,如果这个时候遍历上一节声明的结构体时,修改value,原结构体不会发生任何变化!

for _,v := range tmp{
		v.a = 2
	}

两次输出一致

k:0, v:{1 a}  
k:1, v:{2 b}  
k:0, v:{1 a}  
k:1, v:{2 b}  

for-range接收新值的变量地址都是同一个地址,所以,如果通过(取地址的方式访问)&key或者&value来访问值可能看到不期望的值。

var tmpKey *int
var tmpValue *struct{
    int
    string
}
for k,v := range tmp {
    if k == 0 {
        tmpKey = &k
        tmpValue = &v
    }
}
fmt.Println(*tmpKey)
fmt.Println(*tmpValue)

OutPut

1
{2 b}

解释:这段代码因为在循环中加了if k == 0 的缘故,期望返回0,{1,a},以上的值是不期望的值,因为go编译后的代码会使用一个中间变量存储从切片拷贝的元素值,取地址也是取的这个中间变量的地址。循环执行了两次,k和v的地址不变,导致了数值变化。

1.7.4 编程 Tips #

  • 遍历过程中可以适情况放弃接收 indexvalue,可以一定程度上提升性能
  • 遍历 channel 时,如果 channel 中没有数据,可能会阻塞
  • 尽量避免遍历过程中修改原数据

1.7.5 小结 #

  • range可以用于for 循环中迭代 arrayslicemapchannel字符串所有涉及到遍历输出的东西。
  • for-range 的实现实际上是C风格的for循环
  • 使用index,value接收range返回值会发生一次数据拷贝


本图书由小熊©2021 版权所有,所有文章采用知识署名-非商业性使用-禁止演绎 4.0 国际进行许可。