golang中slice作为参数会怎么样

golang参数传递其实只有一种就是值拷贝,那么slice作为参数传递的时候有什么特别的地方吗?

修改值

我们先看一个小示例

package main

import "fmt"

func changeValue(s []int) {
    fmt.Printf("inner: %v \t%p\n", s, s)
    s[0] = 0
    fmt.Printf("inner: %v \t%p\n", s, s)
}
func main() {
    s1 := []int{1, 2, 3}
    fmt.Printf("outer: %v \t%p\n", s1, s1)
    changeValue(s1)
    fmt.Printf("outer: %v \t%p\n", s1, s1)
}

它输出会是什么样的呢?

outer: [1 2 3]  0xc0000a2140
inner: [1 2 3]  0xc0000a2140
inner: [0 2 3]  0xc0000a2140
outer: [0 2 3]  0xc0000a2140

可以看到 我们通过slice作为参数的函数改变了原slice中的值,而且函数内外slice的内存地址都是一样的。但是golang是值拷贝,它是怎么通过函数内部改变外部的值呢,我们看一下slice的源码实现

type slice struct {
    array unsafe.Pointer
    len   int
    cap   int
}

可以看到,slice是一个结构体,它有三个属性,底层数组指针、长度、容量,因此,虽然值拷贝了一个副本,但是副本中有底层的数组指针,修改的是同一个内存地址,所以在函数内部改变了传入的slice参数,外部也会相应改变。

容量不够,长度增加

我们再看一个demo

package main

import "fmt"

func showAttribute(s []int) {
    fmt.Printf("addr: %p  len: %d  cap: %d  %v\n", s, len(s), cap(s), s)
}
func changeLength(s []int) {
    fmt.Printf("inner: ")
    showAttibute(s)
    s = append(s, 4)
    fmt.Printf("inner: ")
    showAttibute(s)
}

func main() {
    s1 := []int{1, 2, 3}
    fmt.Printf("outer: ")
    showAttibute(s1)
    // changeValue(s1)
    changeLength(s1)
    fmt.Printf("outer: ")
    showAttibute(s1)
}

它输出如下:

outer: addr: 0xc00000a400  len: 3  cap: 3  [1 2 3]
inner: addr: 0xc00000a400  len: 3  cap: 3  [1 2 3]
inner: addr: 0xc00000c330  len: 4  cap: 6  [1 2 3 4]
outer: addr: 0xc00000a400  len: 3  cap: 3  [1 2 3]

可以看到函数内部slice地址发生了变化,这是因为 s1 := []int{1, 2, 3} 建了一个len=3,cap=3的slice,当在后面追加一个值4的时候,slice副本的底层数组重新分配,cap加倍,len加一,因此地址改变。但是副本的变化不会影响函数外部slice的属性。

容量够,长度增加

我们再看另一种情况

func main() {
    s1 := []int{1, 2, 3}
    // fmt.Printf("outer: ")
    // showAttibute(s1)
    // // changeValue(s1)
    // changeLength(s1)
    // fmt.Printf("outer: ")
    // showAttibute(s1)

    s2 := make([]int, 0, 10)
    s2 = append(s2, s1...)
    fmt.Printf("outer: ")
    showAttibute(s2)
    // changeValue(s1)
    changeLength(s2)
    fmt.Printf("outer: ")
    showAttibute(s2)
}

输出如下

outer: addr: 0xc00000e230  len: 3  cap: 10  [1 2 3]
inner: addr: 0xc00000e230  len: 3  cap: 10  [1 2 3]
inner: addr: 0xc00000e230  len: 4  cap: 10  [1 2 3 4]
outer: addr: 0xc00000e230  len: 3  cap: 10  [1 2 3]

与上面不同的是 slice的地址没有改变。这是因为 s2 := make([]int, 0, 10) 建立了一个容量为10的slice,我们给他添加3个数据后,长度变为3,传递到函数 changeLength() 中后,追加了一个4,但是容量>=长度,底层数组没有重新分配,地址不改变,cap不改变,len加一。同样函数内部副本改变的长度信息改变不会影响函数外部slice,因为函数外部slice还是3个值。

range与slice

我们再看一个例子

package main

import "fmt"

func main() {
    arr := [5]int{1, 2, 3, 4, 5}
    sli := []int{1, 2, 3, 4, 5}

    for i, v := range arr { // range只在使用时求值一次,对于数组会拷贝一次,迭代的v来自数组副本
        if i == 0 {
            arr[i+2] = 33 // 在遍历i=0时,修改i=2时的arr值
            fmt.Println(arr)
        }
        arr[i] = v + 100 // 在遍历到i=2时,v是arr副本中的值,之前修改的arr[2]不会影响i=2时的v
    }
    fmt.Println(arr)
    for i, v := range sli { //slice拷贝副本中有底层数组指针,因此修改sli会影响v
        if i == 0 {
            sli[i+2] = 33 // 在遍历i=0时,修改i=2时的sli值
            fmt.Println(sli)
        }
        sli[i] = v + 100 // 遍历到i=2时,v就是sli[2]的值,前面的修改发生作用了
    }
    fmt.Println(sli)
}

其实看了前面的slice传参,这里就很好理解了,range就是个函数,arr和sli都是传进去的参数,虽然都是值拷贝,但是数组拷贝了的数据副本改变不会影响原数组,而slice副本中有底层数组指针,指向同一块内存,修改副本值就是修改原值。它的输出如下:

[1 2 33 4 5]
[101 102 103 104 105]
[1 2 33 4 5]
[101 102 133 104 105]

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