一尘不染

通道是否通过引用隐式传递

go

前往游览的频道示例如下:https :
//tour.golang.org/concurrency/2

package main

import "fmt"

func sum(a []int, c chan int) {
    sum := 0
    for _, v := range a {
        sum += v
    }
    c <- sum // send sum to c
}

func main() {
    a := []int{7, 2, 8, -9, 4, 0}

    c := make(chan int)
    go sum(a[:len(a)/2], c)
    go sum(a[len(a)/2:], c)
    x, y := <-c, <-c // receive from c

    fmt.Println(x, y, x+y)
}

通道c在求和函数中被修改,并且该函数终止后更改仍然存在。显然,c是通过引用传递的,但是没有创建指向c的指针。是否在go中通过引用隐式传递了通道?


阅读 208

收藏
2020-07-02

共1个答案

一尘不染

从技术上讲,它们是被复制的,因为当您使用时make,您是在堆上分配内容,因此从技术上讲,它是幕后的指针。但是指针类型没有公开,因此可以将它们视为引用类型。

编辑 :从规格:

内置函数make采用类型T,该类型必须是切片,映射或通道类型,还可以选择后面是特定于类型的表达式列表。它返回类型T的值(不是*
T)。存储器按照有关初始值的部分中所述进行初始化。

必须先初始化通道,然后才能使用它。Make会这样做,因此可以将其用作引用类型。

这基本上意味着您可以将其传递给函数并对其进行写入或读取。一般的经验法则是,如果你使用makenew或者&,你可以将它传递到另一个功能,无需复制基础数据。

因此,以下是“引用”类型:

  • 切片
  • 地图
  • 频道
  • 指针
  • 功能

传递给函数时,仅复制数据类型(数字,布尔值和结构等)。字符串很特殊,因为它们是不可变的,但不能按值传递。这意味着以下内容将无法正常工作:

type A struct {
    b int
}
func f(a A) {
    a.b = 3
}
func main() {
    s := A{}
    f(s)
    println(s.b) // prints 0
}
2020-07-02