区别
for range :
- 会在遍历之前,先拷贝一份被遍历的数据,然后遍历拷贝的数据
- for range遍历出来元素的地址都是一个,并且与原来元素的地址不同。
循环内增加数组长度
s := []int{0, 1}
for _, v := range s {
s = append(s, v)
}
fmt.Printf("s=%v\n", s)
打印 :
s=[0 1 0 1]
因为range遍历之前已经拷贝了一份数组,所以数组长度即使增加也不影响,同样可以遍历完成。
但如果改成下面这样, 则会造成死循环, 因为数组 s 的长度一直在增加,i 永远都小于数组的长度
s := []int{0, 1}
// 死循环
for i := 0; i < len(s); i++ {
s = append(s, s[i])
}
fmt.Printf("s=%v\n", s)
循环中看元素的地址
s := []int{0, 1}
for i, v := range s {
fmt.Printf("第%d切片元素的地址: %p\n", i, &v)
}
fmt.Println("*************************")
for i := 0; i < len(s); i++ {
fmt.Printf("第%d切片元素的地址: %p\n", i, &s[i])
}
打印:
第0切片元素的地址: 0x14000126008
第1切片元素的地址: 0x14000126008
*************************
第0切片元素的地址: 0x14000126010
第1切片元素的地址: 0x14000126018
可以看到, 使用range的时候,因为拷贝了一份,所以每个元素使用的都是同一个地址,并且与原来元素的地址不同
小结
- 如果需要遍历的 数组/切片/map等 比较大,建议使用原生for循环遍历,因为for range会进行一次拷贝,占用内存。
- 如果在循环中,会给被循环的 数组/切片/map等 增加长度,使用for range,因为长度一直增加,原生for循环会死循环。
- 如果需要使用到元素的地址,使用原生for循环,for range的地址和元素原来的地址不同。
- 理解原理,看情况使用…