Go语言for range(按照键值循环)遍历操作
Go语言可以使用forrange遍历数组、切片、字符串、map及通道(channel)。通过forrange遍历的返回值有一定的规律:
数组、切片、字符串返回索引和值。
map返回键和值。
通道(channel)只返回通道内的值。
遍历数组、切片——获得索引和元素
在遍历代码中,key和value分别代表切片的下标及下标对应的值。下面的代码展示如何遍历切片,数组也是类似的遍历方法:
forkey,value:=range[]int{1,2,3,4}{ fmt.Printf("key:%dvalue:%d\n",key,value) }
代码输出如下:
key:0value:1 key:1value:2 key:2value:3 key:3value:4
遍历字符串——获得字符
Go语言和其他语言类似,可以通过forrange的组合,对字符串进行遍历,遍历时,key和value分别代表字符串的索引(base0)和字符串中的每一个字符。
下面这段代码展示了如何遍历字符串:
varstr="hello你好" forkey,value:=rangestr{ fmt.Printf("key:%dvalue:0x%x\n",key,value) }
代码输出如下:
key:0value:0x68 key:1value:0x65 key:2value:0x6c key:3value:0x6c key:4value:0x6f key:5value:0x20 key:6value:0x4f60 key:9value:0x597d
代码中的v变量,实际类型是rune,实际上就是int32,以十六进制打印出来就是字符的编码。
遍历map——获得map的键和值
对于map类型来说,forrange遍历时,key和value分别代表map的索引键key和索引对应的值,一般被称为map的键值对,因为它们总是一对一对的出现。下面的代码演示了如何遍历map。
m:=map[string]int{ "hello":100, "world":200, } forkey,value:=rangem{ fmt.Println(key,value) }
代码输出如下:
hello100
world200
注意
对map遍历时,遍历输出的键值是无序的,如果需要有序的键值对输出,需要对结果进行排序。
遍历通道(channel)——接收通道数据
forrange可以遍历通道(channel),但是通道在遍历时,只输出一个值,即管道内的类型对应的数据。
下面代码为我们展示了通道的遍历:
c:=make(chanint) gofunc(){ c<-1 c<-2 c<-3 close(c) }() forv:=rangec{ fmt.Println(v) }
代码说明如下:
第1行创建了一个整型类型的通道。
第3行启动了一个goroutine,其逻辑的实现体现在第5~8行,实现功能是往通道中推送数据1、2、3,然后结束并关闭通道。
这段goroutine在声明结束后,在第9行马上被并行执行。
从第11行开始,使用forrange对通道c进行遍历,其实就是不断地从通道中取数据,直到通道被关闭。
在遍历中选择希望获得的变量
在使用forrange循环遍历某个对象时,一般不会同时需要key或者value,这个时候可以采用一些技巧,让代码变得更简单。下面将前面的例子修改一下,参考下面的代码示例:
m:=map[string]int{ "hello":100, "world":200, } for_,value:=rangem{ fmt.Println(value) }
代码输出如下:
100
200
在例子中将key变成了下画线,那么这里的下画线就是匿名变量。什么是匿名变量?
可以理解为一种占位符。
本身这种变量不会进行空间分配,也不会占用一个变量的名字。
在forrange可以对key使用匿名变量,也可以对value使用匿名变量。
再看一个匿名变量的例子:
forkey,_:=range[]int{1,2,3,4}{ fmt.Printf("key:%d\n",key) }
代码输出如下:
key:0 key:1 key:2 key:3
在该例子中,value被设置为匿名变量,只使用key,而key本身就是切片的索引,所以例子输出索引。
我们总结一下for的功能:
Go语言的for包含初始化语句、条件表达式、结束语句,这3个部分均可缺省。
forrange支持对数组、切片、字符串、map、通道进行遍历操作。
在需要时,可以使用匿名变量对forrange的变量进行选取。
补充:学习-go语言坑之forrange
go只提供了一种循环方式,即for循环,在使用时可以像c那样使用,也可以通过forrange方式遍历容器类型如数组、切片和映射。但是在使用forrange时,如果使用不当,就会出现一些问题,导致程序运行行为不如预期。比如,下面的示例程序将遍历一个切片,并将切片的值当成映射的键和值存入,切片类型是一个int型,映射的类型是键为int型,值为*int,即值是一个地址。
packagemain import"fmt" funcmain(){ slice:=[]int{0,1,2,3} myMap:=make(map[int]*int) forindex,value:=rangeslice{ myMap[index]=&value } fmt.Println("=====newmap=====") prtMap(myMap) } funcprtMap(myMapmap[int]*int){ forkey,value:=rangemyMap{ fmt.Printf("map[%v]=%v\n",key,*value) } }
运行程序输出如下:
=====newmap===== map[3]=3 map[0]=3 map[1]=3 map[2]=3
由输出可以知道,不是我们预期的输出,正确输出应该如下:
=====newmap===== map[0]=0 map[1]=1 map[2]=2 map[3]=3
但是由输出可以知道,映射的值都相同且都是3。其实可以猜测映射的值都是同一个地址,遍历到切片的最后一个元素3时,将3写入了该地址,所以导致映射所有值都相同。
其实真实原因也是如此,因为forrange创建了每个元素的副本,而不是直接返回每个元素的引用,如果使用该值变量的地址作为指向每个元素的指针,就会导致错误,在迭代时,返回的变量是一个迭代过程中根据切片依次赋值的新变量,所以值的地址总是相同的,导致结果不如预期。
修正后程序如下:
packagemain import"fmt" funcmain(){ slice:=[]int{0,1,2,3} myMap:=make(map[int]*int) forindex,value:=rangeslice{ num:=value myMap[index]=&num } fmt.Println("=====newmap=====") prtMap(myMap) } funcprtMap(myMapmap[int]*int){ forkey,value:=rangemyMap{ fmt.Printf("map[%v]=%v\n",key,*value) } }
运行程序输出如下:
=====newmap===== map[2]=2 map[3]=3 map[0]=0 map[1]=1
以上为个人经验,希望能给大家一个参考,也希望大家多多支持毛票票。如有错误或未考虑完全的地方,望不吝赐教。