

新闻资讯
技术教程Go中结构体默认浅拷贝,赋值时基本类型字段独立,但slice、map、指针、chan、func和interface{}字段共享底层数据;深拷贝需手动实现Clone方法或谨慎使用序列化。
Go 的结构体是值类型,直接用 = 赋值时,会逐字段复制。如果字段全是基本类型(int、string、bool)或数组,那就真的完全独立;但只要包含指针、slice、map、chan 或 interface{},这些字段的底层数据仍被共享。
以下字段类型在复制后仍与原结构体共用底层数据:
slice:新结构体的 slice 与原结构体指向同一底层数组map:两个结构体的 map 变量引用同一个哈希表*T(任意指针):复制的是地址值,指向同一块内存chan 和 func 类型同理,复制的是引用例如:
type Config struct {
Name string
Tags []string
Data map[string]int
}
cfg1 := Config{
Name: "v1",
Tags: []string{"a", "b"},
Data: map[string]int{"x": 1},
}
cfg2 := cfg1 // 浅拷贝
cfg2.Tags[0] = "z" // cfg1.Tags[0] 也变成 "z"
cfg2.Data["x"] = 99 // cfg1.Data["x"] 也变成 99
没有语言级深拷贝支持,需按需选择策略:
Tags: append([]string(nil), src.Tags...)、Data: copyMap(src.Data)
encoding/gob 或 encoding/json 序列化再反序列化(注意:要求字段
json 不支持 func、chan、循环引用)github.com/jinzhu/copier 或 github.com/mohae/deepcopy,但要注意其对未导出字段、自定义类型的支持限制Clone() 方法,明确控制每个字段如何复制推荐优先手写 Clone(),清晰、可控、无反射开销:
func (c Config) Clone() Config {
clone := c
if c.Tags != nil {
clone.Tags = append([]string(nil), c.Tags...)
}
if c.Data != nil {
clone.Data = make(map[string]int, len(c.Data))
for k, v := range c.Data {
clone.Data[k] = v
}
}
return clone
}
看似方便,但有多个隐性陷阱:
time.Time 会被转成字符串再解析,丢失精度或时区信息(取决于 JSONTime 设置)nil slice 和空 slice 在 JSON 中都变成 [],无法区分func、unsafe.Pointer、sync.Mutex 等会 panic仅适合临时调试或原型阶段,生产代码慎用。
深拷贝的关键不是“有没有工具”,而是清楚每个字段的语义——是该共享还是隔离。结构体设计时就该决定哪些字段用值类型、哪些该封装成带 Copy() 方法的自定义类型,比事后补深拷贝更可靠。