Golang中range的使用方法及注意事项

马谦的博客 · 2018-05-17 09:25:56 · 28072 次点击 · 预计阅读时间 1 分钟 · 大约8小时之前 开始浏览    
这是一个创建于 2018-05-17 09:25:56 的文章,其中的信息可能已经有所发展或是发生改变。

一、用法

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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
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)
}
}

结果:

1
2
3
4
5
6
7
8
0 1
1 2
2 3
0 2
1 3
2 4
Hello 1
World 2

二、注意事项

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

示例一:

1
2
3
4
5
6
7
8
9
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
}

示例二:

1
2
3
4
5
6
7
8
9
10
11
12
13
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迭代遍历引用类型时,底层的数据不会被复制:

1
2
3
4
5
6
7
8
9
10
11
12
13
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
2
[1 200 300]
[101 300 400]

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

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


有疑问加站长微信联系(非本文作者)

本文来自:马谦的博客

感谢作者:马谦的博客

查看原文:Golang中range的使用方法及注意事项

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

28072 次点击  
加入收藏 微博
2 回复  |  直到 2019-12-04 11:44:08
CppToGo
CppToGo · #1 · 5年之前

在对slice进行range的时候,其实每一次都是进行了深拷贝的, 若不进行深拷贝,切片b的输出结果应该为{200,300,103}

package main

import "fmt"

func main() {
    a := [3]int{1, 2, 3}
    b := []int{1, 2, 3}

    for k, v := range a {//v只获取到了a数组中的值
        if k == 0 {
            a[0], a[1] = 100, 200
            fmt.Printf("k:%v a:%v\n", k, a)
        }
        a[k] = 100 + v
    }
    for k, v := range b {//每一次循环v是对b[k]的深拷贝 ,这样才可以解释{101,200,103}的输出结果
        if k == 0 {
            b[0], b[1] = 100, 200
            fmt.Printf("k:%v b:%v\n", k, b)
        }
        b[k] = 100 + v
    }
    fmt.Printf("a: %v\n b: %v\n", a, b)
}

输出结果为:

k:0 a:[100 200 3]
k:0 b:[100 200 3]
    a: [101 102 103]
    b: [101 300 103]
yzqaq
yzqaq · #2 · 5年之前

切片不是深拷贝,你的例子只表明 range 数组时一下子把所有值都读取了,而range 切片时是一次循环读一个。 详细的说,数组时 v的值依次是 1,2,3 ,而切片时 v的值依次是 1 200 3.

package main

import (
    "fmt"
)

func main() {
    type temp struct {
        field1 string
        field2 int
    }
    temps := []*temp{
        {
            field1: "a",
            field2: 1,
        },
        {
            field1: "b",
            field2: 2,
        },
        {
            field1: "c",
            field2: 3,
        },
    }
    for _, v := range temps {
        v.field2 = 4
    }
    fmt.Printf("%v %v %v", *temps[0], *temps[1], *temps[2])//{a 4} {b 4} {c 4}

}
    temps2 := []map[int]string{
        map[int]string{1: "a"},
        {2: "b"}, {3: "c"},
    }
    for _, v := range temps2 {
        v[4] = "d"
    }
    fmt.Printf("%v",temps2)//[map[1:a 4:d] map[2:b 4:d] map[3:c 4:d]] 对v更改后temps2 也被更改了
添加一条新回复 (您需要 登录 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传