关于go:GO-的-range-如何使用

46次阅读

共计 2199 个字符,预计需要花费 6 分钟才能阅读完成。

GO 语言的 for…range 能做什么呢?

for…range 如何应用 ?

for…range 的返回值有哪些状况,能够对于任何数据结构应用吗?

for…range 的返回值如果不须要,能够如何解决?

for…range 的数据是如何传递的?

刚学习 golang 的 xdm 会不会有下面的疑难?其实很简略,咱们就一个一个的来分享和实操一遍

GO 语言的 for…range 能做什么呢?

golang 的 for…range 是 go 本身的语法,能够用来遍历数据结构,有如下数据结构能够遍历

  • 切片 slice
  • 数组 array
  • map 哈希表
  • channel 通道

for…range 如何应用 ?

咱们别离来看看能够如何应用他们,for…range 相当于一个迭代器,能够遍历数据结构的键 / 索引 和值

数组 array

  • 初始化一个数组
  • 应用 for…range 遍历,对应的是 索引 和 值
func main() {myArray := [5]int{1, 2, 3, 4, 5}

    for i, v := range myArray {fmt.Printf("%d -- %d -- %p\n", i, v, &v)
    }
}

切片 slice

  • 初始化一个切片
  • 应用 for…range 遍历,对应的是 索引 和 值
mySlice := []int{1, 2, 3, 4, 5}
    
for i, v := range mySlice {fmt.Printf("%d -- %d -- %p\n", i, v, &v)
}

map 哈希表

  • 初始化一个 map 哈希表
  • 应用 for…range 遍历,map 对应的 键值对
myMap := map[string]string{
    "name":  "xmt",
    "hobby": "program",
    "addr":  "mengli",
}
for k, v := range myMap {fmt.Printf("%s -- %s -- %p\n", k, v, &v)
}

channel 通道

  • 创立一个能够缓冲 10 个 int 类型数据的通道
  • 创立一个协程专门向通道中写入数据
  • 主协程遍历通道,读取数据

package main

import "fmt"

var myCh = make(chan int, 10)

func writeCh() {
    for i := 0; i < 5; i++ {myCh <- i}
    close(myCh)

}
func main() {go writeCh()
    for {
        for data := range myCh {fmt.Println(data)
        }
        break
    }

}

for…range 的返回值有哪些状况,能够对于任何数据结构应用吗?

并不是所有数据结构都能够应用 for…range 的,如下构造能够应用这个办法

返回值 1 返回值 2 数据传递
字符串 索引 索引对应的值 值传递
数组或者切片 索引 索引对应的值 数组:值传递
切片:援用传递
哈希表 键对应的值 指针
通道 通道中的数据 指针

for…range 的返回值如果不须要,能够如何解决?

置信写过 golang 的 xdm 都晓得,go 外面对于咱们不须要的返回值,都能够应用 - 来示意,因而 for…range 当然也能够这样

例如:

myMap := map[string]string{
    "name":  "xmt",
    "hobby": "program",
    "addr":  "mengli",
}
for _, v := range myMap {fmt.Printf("%s -- %p\n", v, &v)
}

for…range 的数据是如何传递的?

数据都是通过复制传递的,也就是都是值传递的,只不过因为数据结构不一样,数组和切片在这里还是有一些区别的

后面的文章,咱们有说到,切片,对应着 有 3 个元素的数据结构,一个是 cap,一个是 len,一个是 ptr,指向的是一个底层数组

切片是援用传递,然而传递数据的时候,切片这个变量是值传递,然而它理论指向的底层数组还是不会变的

咱们来写一个 demo 查看一下:

咱们思路是这个样子:

遍历一个 数组 / 切片,在遍历以后值的时候,去批改前面索引对应的值,全副遍历结束后,查看后果中,理论的数据是否会被批改掉,若批改掉了,则是切片是传援用,若没有批改掉,则是数组是传值

数组的成果

myArray := [5]int{1, 2, 3, 4, 5}
fmt.Println(myArray)

for i, v := range myArray {
    if  i == 0{myArray[2] = 888
    }
    fmt.Printf("%d -- %d -- %p\n", i, v, &myArray[i])
}

成果如下:

go run main.go
[1 2 3 4 5]
0 -- 1 -- 0x1189c120
1 -- 2 -- 0x1189c124
2 -- 3 -- 0x1189c128
3 -- 4 -- 0x1189c12c
4 -- 5 -- 0x1189c130

切片的成果

mySlice := []int{1, 2, 3, 4, 5}
fmt.Println(mySlice)
for i, v := range mySlice {
    if i == 0{mySlice[2] = 888
    }
    fmt.Printf("%d -- %d -- %p\n", i, v, &mySlice[i])
}

成果如下:

go run main.go
[1 2 3 4 5]
0 -- 1 -- 0x1140e340
1 -- 2 -- 0x1140e344
2 -- 888 -- 0x1140e348
3 -- 4 -- 0x1140e34c
4 -- 5 -- 0x1140e350

通过上述的案例,置信心里都有点谱了吧

欢送点赞,关注,珍藏

敌人们,你的反对和激励,是我保持分享,提高质量的能源

好了,本次就到这里

技术是凋谢的,咱们的心态,更应是凋谢的。拥抱变动,背阴而生,致力向前行。

我是 阿兵云原生,欢送点赞关注珍藏,下次见~

正文完
 0