golang for range的坑

本文深入探讨了Go语言中range函数的使用方式,特别是在处理非引用类型和引用类型时的行为差异。通过具体案例,展示了range如何在数组和切片中工作,以及在遍历过程中变量的地址和值的变化。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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}
 */
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值