case 1: 非引用类型
range会复制对象,而不是不是直接在原对象上操作。
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]
}
case 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) //[101 300 400]
}
Note:
for,range 的k,v变量在整个遍历过程中共用,不能直接进行引用传递,即地址传递。
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]
}
fmt.Printf("i:{addr: %p; value: %v}, v:{addr: %p; value: %v}\n", &i, i, &v, v)
a[i] = v + 100
}
fmt.Println(a)
}
// 注意,循环中i, v的地址是一样的,也就是说这个变量是被复用的!!禁止直接使用i或者v的地址!!!
/* result:
i:{addr: 0xc000054080; value: 0}, v:{addr: 0xc000054088; value: 1}
i:{addr: 0xc000054080; value: 1}, v:{addr: 0xc000054088; value: 200}
i:{addr: 0xc000054080; value: 2}, v:{addr: 0xc000054088; value: 300}
*/