来自公众号:新世界杂货铺

循环永动机

Q1: 如果咱们在遍历数组的同时批改数组元素,是否失去一个永远都不会进行的循环呢?

func main() { arr := []int{1, 2, 3} for _, v := range arr {     arr = append(arr, v) } fmt.Println(arr)}// 输入: 1 2 3 1 2 3

上述代码的输入意味着循环只遍历了原始切片中的三个元素,咱们遍历切片时追加的元素不会减少循环的执行次数, 所以循环最终还是停了下来

答: 对于所有的range循环, Go语言会在编译期间将原切片或者数组赋值给一个新的变量ha, 在赋值的过程中就产生了拷贝, 所以咱们遍历的切片曾经不是原始的切片变量了

神奇的指针

Q2: 咱们遍历一个数组时,如果获取range返回变量的地址并保留到另一个数组或哈希时, 就会遇到令人困惑的景象

func main() {    arr := []int{1, 2, 3}    newArr := []*int{}    for _, v := range arr {        newArr = append(newArr, &v)    }    for _, v := range newArr {        fmt.Println(*v)    }}// 输入: 3 3 3

答: 遇到这种同时遍历索引和元素的range循环时, go语言会额定创立一个新的v2变量存储切片中的元素, 循环中应用的这个变量v2会在每一次迭代被从新赋值而笼罩, 在赋值时也产生了拷贝. 因为在循环中返回的变量的地址都完全相同, 所以才会呈现神奇的指针的景象

map的随机遍历

Q3: go语言中应用range遍历哈希表时, 往往都会失去不同的后果?

答: 但这并不是阐明哈希表不稳固, 这是go语言成心这样设计的, 他在运行时为哈希表遍历引入不确定性, 也是通知所有应用go语言的使用者, 程序不要依赖于哈希表的稳固遍历。

参考

https://draveness.me/golang/docs/part2-foundation/ch05-keyword/golang-for-range/