浅拷贝复制字段值但共享引用数据,修改可能相互影响;深拷贝递归复制所有层级,完全独立。Go中需手动或通过gob、第三方库实现深拷贝。

在Go语言中,深拷贝和浅拷贝的区别主要体现在对复合类型(如切片、map、指针、结构体等)复制时是否共享底层数据。
浅拷贝:只复制值,但共享底层引用数据
浅拷贝是指复制对象的字段值,如果字段是引用类型(如指针、slice、map、chan等),则复制的是这些引用的“副本”,它们仍指向同一个底层数据。修改其中一个,可能会影响另一个。
常见的浅拷贝操作包括:
- 直接赋值结构体(包含 slice、map 等字段)
- 使用 map[string]interface{} = srcMap 复制 map
- 切片的截取操作:b := a[1:3],a 和 b 共享底层数组
type User struct {
Name string
Tags []string
}
u1 := User{Name: "Alice", Tags: []string{"go", "dev"}}
u2 := u1 // 浅拷贝
u2.Tags[0] = "rust"
// u1.Tags[0] 也会变成 "rust"
深拷贝:完全独立复制所有层级数据
深拷贝不仅复制对象本身,还会递归复制所有引用类型的底层数据,使得源对象和目标对象完全独立,互不影响。
立即学习“go语言免费学习笔记(深入)”;
Go 没有内置的通用深拷贝函数,需要手动实现或借助第三方库。
实现方式包括:
- 手动逐字段复制:适用于结构清晰的小结构体
- 使用 encoding/gob 序列化反序列化:可实现通用深拷贝,但要求类型可导出且支持 gob
- 使用第三方库:如 github.com/mohae/deepcopy 或 google.golang.org/protobuf/proto(针对 proto 对象)
func DeepCopy(dst, src interface{}) error {
var buf bytes.Buffer
enc := gob.NewEncoder(&buf)
dec := gob.NewDecoder(&buf)
if err := enc.Encode(src); err != nil {
return err
}
if err := dec.Decode(dst); err != nil {
return err
}
return nil
}
// 使用
u1 := User{Name: "Bob", Tags: []string{"web"}}
var u2 User
DeepCopy(&u2, &u1)
u2.Tags[0] = "api"
// u1.Tags 不受影响
如何判断是否需要深拷贝?
当你希望两个变量完全独立,尤其是涉及并发读写、长期持有副本或修改副本不应影响原数据时,就需要深拷贝。
常见需要注意的类型:
- 结构体中含有 slice、map、指针字段
- 嵌套结构体中有引用类型
- 在 goroutine 中传递副本但仍想避免数据竞争
基本上就这些。理解拷贝行为的关键是看“引用类型字段”是否共用底层数据。简单赋值通常是浅拷贝,深拷贝需额外处理。










