Go结构体默认浅拷贝:值类型字段独立复制,引用类型字段共享底层数据;需深拷贝时应依场景选手动复制、gob序列化或第三方库,并注意未导出字段、channel等限制。

Go 语言中结构体拷贝默认是浅拷贝,但是否“真正独立”,取决于结构体字段的类型。值类型字段(如 string、int、array、不含引用字段的 struct)会被完整复制;而引用类型字段(如 slice、map、pointer、func)只复制地址,新旧结构体共享底层数据 —— 这就是问题根源。
结构体赋值默认是浅拷贝
直接用 = 赋值或函数传值时,Go 对结构体做的是逐字段复制:
- 值类型字段:内存内容被复制一份,互不影响
- 引用类型字段:仅复制指针/头信息,底层数组、哈希表或对象仍共用
例如:
type User struct {
Name string
Tags []string // slice 是引用类型
}
u1 := User{Name: "Alice", Tags: []string{"go", "dev"}}
u2 := u1 // 浅拷贝
u2.Tags[0] = "rust" // 修改影响 u1
// 此时 u1.Tags[0] 也变成 "rust"
什么时候需要深拷贝
当结构体含引用字段,且你希望副本完全隔离、修改不波及原结构时,必须深拷贝。典型场景包括:
立即学习“go语言免费学习笔记(深入)”;
实现深拷贝的常用方式
没有银弹,按场景选合适方法:
- 手动逐字段复制:适合简单结构体,对 slice/map/pointer 字段显式创建新实例并拷贝内容
- encoding/gob 序列化:通用、无需额外依赖,但要求类型支持 gob(不能含 unexported func/channel/unsafe.Pointer),性能较低
-
第三方库:如
github.com/mohae/deepcopy或github.com/jinzhu/copier,自动处理嵌套与指针,适合复杂结构
gob 示例:
import "bytes"
import "encoding/gob"
func DeepCopy(src, dst interface{}) error {
var buf bytes.Buffer
enc := gob.NewEncoder(&buf)
dec := gob.NewDecoder(&buf)
if err := enc.Encode(src); err != nil {
return err
}
return dec.Decode(dst)
}
// 使用
var u2 User
err := DeepCopy(&u1, &u2)
容易忽略的关键点
深拷贝不是“只要用了就安全”:
- 若结构体字段含未导出字段(首字母小写),gob 和多数反射类库无法访问,会导致拷贝失败或字段为零值
- 包含 channel、func、unsafe.Pointer 的结构体无法安全深拷贝,应避免在需拷贝的结构中使用
- 循环引用结构体(如树节点含 parent 指针)用反射深拷贝可能 panic,需特殊处理










