go slice与函数传值的理解

go语言中所有的传值方式都是传值操作。

今天遇到了以下代码:

func main(){
      slice := make([]int ,1,1)
     fmt.Println(slice)
     change(s)
     fmt.Println(slice)  
}

func change(s []int){
     s = append(s,2) 
} 

输出结果是:

    0

    0

当时感到有些疑惑,学c或c++时,传入的参数为指针(数组首地址)时,change内这种直接修改参数s方法时可以修改值的,那么为何在go中无效呢?

1.go语言中,所有的参数传递都是值传递。

什么意思?就是在函数传递值时,都会拷一个副本传递给函数进行操作,就算是引用类型或者指针也不例外。

例如:

func main(){
        st := make([]int, 1, 1)
    fmt.Printf("st 原始切片的地址为:%p\n", &st)
    SliceTest(st)
}

func SliceTest(s []int) {
    fmt.Printf("st 传入后参数s的地址为:%p\n", &s)
    s = append(s, 3)
}

在我的电脑上的输出为:

st 原始切片的地址为:0xc000004540
st 传入后参数s的地址为:0xc000004560

可以看到,go语言中仅仅拷贝了st的一个副本进入SliceTest函数中进行操作,并未操作st本身。

2.go语言的切片,实际上是指向一个底层的数组。

在进入SliceTest函数时,s和st指向的底层数组都一样。

但是,经过append的操作,s原本指向的底层数组不满足追加元素的条件,所以要开辟一个新的数组来进行追加操作。

于是,s会指向一个新的底层数组。

但是这些操作和原切片st完全无关,因为s仅仅是st的副本,它们的地址不一样,s指向了新的数组,st并未改变,所以就有了开篇那段代码的结果。

对go的传值理解进一步加深。

猜你喜欢

转载自www.cnblogs.com/xuchen950317/p/12208732.html