值类型通过channel传递时是复制值,接收方修改不影响发送方;指针传递则是地址,双方共享同一数据。

在Go语言中,channel用于goroutine之间的通信,传递数据时值类型和指针的行为有明显差异。理解这些差异对编写高效、安全的并发程序至关重要。
值类型通过channel传递的是副本
当使用值类型(如int、string、struct等)进行channel传输时,Go会复制整个值。这意味着接收方拿到的是原始数据的一份拷贝,对接收值的修改不会影响发送方的数据。
例如:
func main() {c := make(chan struct{ X int })
go func() {
v := v.X = 10 // 修改的是副本
}()
s := struct{ X int }{X: 5}
c time.Sleep(time.Second)
fmt.Println(s.X) // 输出仍是5
}
这种机制保证了数据安全,但也可能带来性能开销,特别是结构体较大时。
立即学习“go语言免费学习笔记(深入)”;
指针通过channel传递的是地址引用
当channel传递指针时,实际传输的是变量的内存地址。接收方可以通过该指针直接修改原始数据。
示例:
func main() {c := make(chan *struct{ X int })
go func() {
p := p.X = 10 // 直接修改原数据
}()
s := &struct{ X int }{X: 5}
c time.Sleep(time.Second)
fmt.Println(s.X) // 输出10
}
这种方式避免了大对象复制,提升性能,但需要小心竞态条件,多个goroutine可能同时修改同一数据。
选择依据:性能与安全性权衡
是否使用指针传递取决于具体场景:
- 小对象或基础类型:推荐传值,开销小且天然线程安全
- 大结构体或需修改原数据:传指针更高效,但需配合锁或其他同步机制确保安全
- 不可变数据:传值更直观,语义清晰
- 频繁传递同一对象:指针可减少内存分配和GC压力
基本上就这些。关键在于明确你希望共享数据还是隔离数据,再结合性能需求做决定。不复杂但容易忽略细节。










