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]
有疑问加站长微信联系(非本文作者)