如果通过Go中的通道发送大型结构,它是否实际在goroutines之间复制?
例如,在下面的代码中,Go会实际复制goroutines生产者和消费者之间的所有largeStruct数据吗?
package main
import (
"fmt"
"sync"
)
type largeStruct struct {
buf [10000]int
}
func main() {
ch := make(chan largeStruct)
wg := &sync.WaitGroup{}
wg.Add(2)
go consumer(wg, ch)
go producer(wg, ch)
wg.Wait()
}
func producer(wg *sync.WaitGroup, output chan<- largeStruct) {
defer wg.Done()
for i := 0; i < 5; i++ {
fmt.Printf("producer: %d\n", i)
output <- largeStruct{}
}
close(output)
}
func consumer(wg *sync.WaitGroup, input <-chan largeStruct) {
defer wg.Done()
i := 0
LOOP:
for {
select {
case _, ok := <-input:
if !ok {
break LOOP
}
fmt.Printf("consumer: %d\n", i)
i++
}
}
}
答案 0 :(得分:16)
是的,一切都是Go中的副本,您可以通过更改频道使用指针(又名chan *largeStruct
)轻松解决这个问题。
// demo:http://play.golang.org/p/CANxwt8s2B
如您所见,指向v.buf
的指针在每种情况下都不同,但如果将其更改为chan *largeStruct
,指针将是相同的。
@LucasJones提供了一个更容易理解的示例:https://play.golang.org/p/-VFWCgOnh0
正如@nos指出的那样,如果你在发送它们后修改两个goroutine中的值,那么就会有潜在的竞争。
答案 1 :(得分:5)
The Go Programming Language Specification
send语句在通道上发送值。频道表达 必须是通道类型,通道方向必须允许发送 操作,以及要发送的值的类型必须可分配给 频道的元素类型。
它是一个副本,因为通过分配到频道的元素类型将值发送到频道。如果值是结构,则复制结构。如果值是指向结构的指针,则复制指向结构的指针。