go语言中指针本身不能实现深拷贝,仅复制地址;深拷贝需递归、反射(如reflect.value.clone())或序列化,且须处理指针、切片、map、struct四类复合类型及字段导出性。

Go 语言中,指针本身不能直接实现深拷贝——它只是地址引用,对指针解引用后赋值仍是浅拷贝;真要深拷贝,得靠递归、反射或序列化,指针只是辅助手段,不是解决方案。
为什么直接复制指针不是深拷贝
把一个结构体指针赋给另一个变量,两者指向同一块内存。修改 *p2 就等于修改 *p1,根本没产生新数据副本。
常见错误现象:
- 原结构体字段改了,副本“同步”变了
-
json.Marshal(json.Unmarshal(...))看似深拷贝,但忽略未导出字段、不支持函数/通道等类型 - 用
reflect.Copy复制指针字段时,只复制了指针值(即地址),不是目标值
用 reflect.DeepCopy 实现带指针字段的深拷贝
标准库没有 reflect.DeepCopy,但可用 reflect.Value.DeepCopy(Go 1.21+)或手写递归逻辑。注意:必须处理指针、切片、map、struct 四类复合类型,且所有字段需可导出(首字母大写)。
立即学习“go语言免费学习笔记(深入)”;
实操建议:
- 优先用 Go 1.21+ 的
reflect.Value.Clone(),它会深拷贝底层数据,包括指针指向的内容(前提是被指向值本身可被反射访问) - 对含
*int、*string等基础类型指针字段的 struct,Clone()会新建指针并指向新分配的值 - 遇到
unsafe.Pointer、func、chan、map[interface{}]等,Clone()会 panic,必须提前过滤或降级处理
手动递归深拷贝时指针字段怎么处理
手写时最容易漏掉的是「指针字段的双重解引用与重分配」:不能只 new 一个新指针,还要确保它指向的新值是原值的深拷贝。
示例关键逻辑:
if v.Kind() == reflect.Ptr {
if v.IsNil() {
return reflect.Zero(v.Type())
}
elem := v.Elem()
copiedElem := deepCopyValue(elem) // 递归拷贝所指对象
ptr := reflect.New(copiedElem.Type())
ptr.Elem().Set(copiedElem)
return ptr
}
要点:
- 空指针(
nil)要保留为nil,不能强制new -
reflect.New()创建新指针,.Elem().Set()把深拷贝后的值塞进去 - 若原指针指向 interface{},需先判断其动态类型再分发,否则会丢失类型信息
性能与兼容性取舍:什么时候不该用反射深拷贝
反射深拷贝在运行时开销大,且无法静态检查字段可访问性。生产环境高频调用场景应规避。
替代方案选择依据:
- 结构体固定且简单 → 手写
Clone()方法,显式 new + 字段逐个深拷贝(如对slice用append([]T(nil), s...)) - 含大量嵌套或未知结构 → 用
gob编码再解码(比 JSON 更准,支持 unexported 字段,但要求类型注册) - 需要跨进程或持久化 → JSON/YAML 序列化是更安全的选择,但明确放弃对指针语义和部分类型的保真
- 第三方库如
copier或deepcopy可减少样板代码,但要注意它们对循环引用的处理策略(多数直接 panic)
最易被忽略的一点:深拷贝后,原对象中通过指针共享的状态(比如某个 *sync.Mutex)会被复制成两个独立锁,这未必是预期行为——深拷贝不只是技术操作,更是语义决策。










