4

我正在尝试制作一个非常简单的程序来修改数组,但是如果我将它们转换为类型,则会遇到一些有趣的行为。https://play.golang.org/p/KC7mqmHuLw看来,如果我有一个数组,则按引用传递,但如果我有一个类型,则按值传递。它是否正确?

我有两个变量 b 和 c,它们都是 3 个整数的数组,但 c 是 cT 类型,在其他所有方面它们应该是相同的。我可以将值分配为b[0]=-1c[0]=-1,但如果我将这些数组作为参数传递给函数,它们的行为就会大不相同。

程序的输出是:

b 之前:[1 2 3]

在 c 之前:[1 2 3]

*在 b 之后:[-1 2 0]

*在 c 之后:[-1 2 3]

*什么?c: [-1 2 0]

我最初的假设是“b 之后”和“c 之后”的行应该是相同的。我做错了什么还是我对按值传递给函数的类型正确(即在传递给函数之前创建变量的副本)?

package main

import "fmt"

type cT [3]int

func main() {
    b := []int{1, 2, 3}
    c := cT{1, 2, 3}

    fmt.Println("before b:", b)
    fmt.Println("before c:", c)

    b[0] = -1
    c[0] = -1
    mangleB(b) // ignore return value
    mangleC(c) // ignore return value

    fmt.Println("*after b:", b)
    fmt.Println("*after c:", c)

    c = mangleC(c)    
    fmt.Println("*what? c:", c)    
}

func mangleB(row []int) []int {
    row[2] = 0
    return row
}

func mangleC(row cT) cT{
    row[2] = 0
    return row
}
4

2 回答 2

4

Go 编程语言规范

数组类型

数组是单一类型的元素的编号序列,称为元素类型。

切片类型

切片是底层数组的连续段的描述符,并提供对该数组中元素的编号序列的访问。

来电

在函数调用中,函数值和参数按通常的顺序计算。在它们被评估之后,调用的参数按值传递给函数,被调用的函数开始执行。函数的返回参数在函数返回时按值传回调用函数。


type cT [3]int

b := []int{1, 2, 3}
c := cT{1, 2, 3}

我有两个变量,b并且c都是 3 个整数的数组


不,你没有!

b是长度 ( ) 3 和容量 ( ) 3的切片,是( ) 3的数组intlen(b)cap(b)clen(c)int

在 Go 中,所有参数都是按值传递的。b作为切片描述符c传递,作为数组传递。切片描述符是struct具有切片长度和容量以及指向底层数组的指针。

于 2017-11-25T13:52:12.623 回答
0

看评论:

func main() {
    b := []int{1, 2, 3} // slice
    c := cT{1, 2, 3} // array

    fmt.Println("before b:", b) 
    fmt.Println("before c:", c) 

    b[0] = -1
    c[0] = -1

    // passing in a slice which you can think of as ref to array
    // pass by value, and it is copy of ref to array 
    mangleB(b) // ignore return value

    // passing in copy of array (pass by value)
    // yes full shallow copy of array 
    mangleC(c) // ignore return value
    // if you ignore return modifications are lost

    fmt.Println("*after b:", b)
    fmt.Println("*after c:", c)

    // return value is modified array
    c = mangleC(c)
    // c now copy of array from line 24

    fmt.Println("*what? c:", c)
}

https://play.golang.org/p/OAaCMhc-Ug

于 2017-11-25T14:27:28.327 回答