当前位置 博文首页 > Go语言for range(按照键值循环)遍历操作

    Go语言for range(按照键值循环)遍历操作

    作者:wilsonyx 时间:2021-02-09 06:28

    Go 语言可以使用 for range 遍历数组、切片、字符串、map 及通道(channel)。通过 for range 遍历的返回值有一定的规律:

    数组、切片、字符串返回索引和值。

    map 返回键和值。

    通道(channel)只返回通道内的值。

    遍历数组、切片——获得索引和元素

    在遍历代码中,key 和 value 分别代表切片的下标及下标对应的值。下面的代码展示如何遍历切片,数组也是类似的遍历方法:

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

    代码输出如下:

    key:0 value:1
    key:1 value:2
    key:2 value:3
    key:3 value:4
    

    遍历字符串——获得字符

    Go 语言和其他语言类似,可以通过 for range 的组合,对字符串进行遍历,遍历时,key 和 value 分别代表字符串的索引(base0)和字符串中的每一个字符。

    下面这段代码展示了如何遍历字符串:

    var str = "hello 你好"
    for key, value := range str {
      fmt.Printf("key:%d value:0x%x\n", key, value)
    }

    代码输出如下:

    key:0 value:0x68
    key:1 value:0x65
    key:2 value:0x6c
    key:3 value:0x6c
    key:4 value:0x6f
    key:5 value:0x20
    key:6 value:0x4f60
    key:9 value:0x597d
    

    代码中的 v 变量,实际类型是 rune,实际上就是 int32,以十六进制打印出来就是字符的编码。

    遍历map——获得map的键和值

    对于 map 类型来说,for range 遍历时,key 和 value 分别代表 map 的索引键 key 和索引对应的值,一般被称为 map 的键值对,因为它们总是一对一对的出现。下面的代码演示了如何遍历 map。

    m := map[string]int{
      "hello": 100,
      "world": 200,
    }
    for key, value := range m {
      fmt.Println(key, value)
    }

    代码输出如下:

    hello 100

    world 200

    注意

    对 map 遍历时,遍历输出的键值是无序的,如果需要有序的键值对输出,需要对结果进行排序。

    遍历通道(channel)——接收通道数据

    for range 可以遍历通道(channel),但是通道在遍历时,只输出一个值,即管道内的类型对应的数据。

    下面代码为我们展示了通道的遍历:

    c := make(chan int)
    go func() {
      c <- 1
      c <- 2
      c <- 3
      close(c)
    }()
    for v := range c {
      fmt.Println(v)
    }

    代码说明如下:

    第 1 行创建了一个整型类型的通道。

    第 3 行启动了一个 goroutine,其逻辑的实现体现在第 5~8 行,实现功能是往通道中推送数据 1、2、3,然后结束并关闭通道。

    这段 goroutine 在声明结束后,在第 9 行马上被并行执行。

    从第 11 行开始,使用 for range 对通道 c 进行遍历,其实就是不断地从通道中取数据,直到通道被关闭。

    在遍历中选择希望获得的变量

    在使用 for range 循环遍历某个对象时,一般不会同时需要 key 或者 value,这个时候可以采用一些技巧,让代码变得更简单。下面将前面的例子修改一下,参考下面的代码示例:

    m := map[string]int{
      "hello": 100,
      "world": 200,
    }
    for _, value := range m {
      fmt.Println(value)
    }

    代码输出如下:

    100

    200

    在例子中将 key 变成了下画线,那么这里的下画线就是匿名变量。什么是匿名变量?

    可以理解为一种占位符。

    本身这种变量不会进行空间分配,也不会占用一个变量的名字。

    在 for range 可以对 key 使用匿名变量,也可以对 value 使用匿名变量。

    再看一个匿名变量的例子:

    for key, _ := 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 个部分均可缺省。

    for range 支持对数组、切片、字符串、map、通道进行遍历操作。

    在需要时,可以使用匿名变量对 for range 的变量进行选取。

    补充:学习-go语言坑之for range

    go只提供了一种循环方式,即for循环,在使用时可以像c那样使用,也可以通过for range方式遍历容器类型如数组、切片和映射。但是在使用for range时,如果使用不当,就会出现一些问题,导致程序运行行为不如预期。比如,下面的示例程序将遍历一个切片,并将切片的值当成映射的键和值存入,切片类型是一个int型,映射的类型是键为int型,值为*int,即值是一个地址。

    package main 
    import "fmt" 
    func main() {
      slice := []int{0, 1, 2, 3}
      myMap := make(map[int]*int)
     
      for index, value := range slice {
        myMap[index] = &value
      }
      fmt.Println("=====new map=====")
      prtMap(myMap)
    }
     
    func prtMap(myMap map[int]*int) {
      for key, value := range myMap {
        fmt.Printf("map[%v]=%v\n", key, *value)
      }
    }

    运行程序输出如下:

    =====new map=====
    map[3]=3
    map[0]=3
    map[1]=3
    map[2]=3
    

    由输出可以知道,不是我们预期的输出,正确输出应该如下:

    =====new map=====
    map[0]=0
    map[1]=1
    map[2]=2
    map[3]=3

    但是由输出可以知道,映射的值都相同且都是3。其实可以猜测映射的值都是同一个地址,遍历到切片的最后一个元素3时,将3写入了该地址,所以导致映射所有值都相同。

    其实真实原因也是如此,因为for range创建了每个元素的副本,而不是直接返回每个元素的引用,如果使用该值变量的地址作为指向每个元素的指针,就会导致错误,在迭代时,返回的变量是一个迭代过程中根据切片依次赋值的新变量,所以值的地址总是相同的,导致结果不如预期。

    修正后程序如下:

    package main 
    import "fmt" 
    func main() {
      slice := []int{0, 1, 2, 3}
      myMap := make(map[int]*int)
     
      for index, value := range slice {
        num := value
        myMap[index] = &num
      }
      fmt.Println("=====new map=====")
      prtMap(myMap)
    }
     
    func prtMap(myMap map[int]*int) {
      for key, value := range myMap {
        fmt.Printf("map[%v]=%v\n", key, *value)
      }
    }

    运行程序输出如下:

    =====new map=====
    map[2]=2
    map[3]=3
    map[0]=0
    map[1]=1

    以上为个人经验,希望能给大家一个参考,也希望大家多多支持站长博客。如有错误或未考虑完全的地方,望不吝赐教。

    js
    下一篇:没有了