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]
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,657评论 6 505
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,889评论 3 394
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,057评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,509评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,562评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,443评论 1 302
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,251评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,129评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,561评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,779评论 3 335
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,902评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,621评论 5 345
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,220评论 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,838评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,971评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,025评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,843评论 2 354

推荐阅读更多精彩内容