一、用法

range类似迭代器,可以遍历数组,字符串,map等等,对象的不同,返回的结果也不同。

package main
import "fmt"
func main(){
    //数组的遍历
    a := [3]int {1, 2, 3}
    for i, n := range a{
        fmt.Println(i, n)
    }
    //切片的遍历
    b := []int{2, 3, 4}
    for i, n := range b{
        fmt.Println(i, n)
    }
    //map的遍历
    c := map[string]int{"Hello":1, "World":2}
    for k, v := range c{
        fmt.Println(k, v)
    }
}

结果:

0 1
1 2
2 3
0 2
1 3
2 4
Hello 1
World 2

二、注意事项

1. range会复制对象,而不是不是直接在原对象上操作。

示例一:

package main
import "fmt"
func main(){
    a := [3]int {1, 2, 3}
    for _, v := range a{ //复制一份a遍历[1, 2, 3]
        v += 100 //v是复制对象中的值,不会改变a数组元素的值
    }
    fmt.Println(a) //1 2 3
}

示例二:

package main
import "fmt"
func main(){
    a := [3]int {1, 2, 3}
    for i, v := range a{ //i,v从a复制的对象里提取出
        if i == 0{
            a[1], a[2] = 200, 300
            fmt.Println(a) //输出[1 200 300]
        }
        a[i] = v + 100 //v是复制对象里的元素[1, 2, 3]
    }
    fmt.Println(a)  //输出[101, 102, 103]
}

2. 使用range迭代遍历引用类型时,底层的数据不会被复制:

package main
import "fmt"
func main(){
    a := []int {1, 2, 3} //改成slice
    for i, v := range a{ 
        if i == 0{
            a[1], a[2] = 200, 300
            fmt.Println(a) //[1 200 300]
        }
        a[i] = v + 100 
    }
    fmt.Println(a)  
}

结果:

[1 200 300]
[101 300 400]

因为切片的内部结构为struct slice{*point, len, cap}。

数据部分是一个指针,指向地址,复制对象的时候只是把指针的值复制了,而不是重新拷贝一块新的内存再把值放进去,所以修改的时候还是修改的原来的值,和C++里的浅拷贝一样。

最后修改:2017 年 08 月 20 日
喜欢就给我点赞吧