浅拷贝复制字段值但共享引用数据,修改可能相互影响;深拷贝递归复制所有层级,完全独立。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 对象)gob 实现深拷贝示例:
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 UserDeepCopy(&u2, &u1)u2.Tags[0] = "api"// u1.Tags 不受影响
如何判断是否需要深拷贝?
当你希望两个变量完全独立,尤其是涉及并发读写、长期持有副本或修改副本不应影响原数据时,就需要深拷贝。
常见需要注意的类型:
结构体中含有 slice、map、指针字段 嵌套结构体中有引用类型 在 goroutine 中传递副本但仍想避免数据竞争
基本上就这些。理解拷贝行为的关键是看“引用类型字段”是否共用底层数据。简单赋值通常是浅拷贝,深拷贝需额外处理。
以上就是Golang深拷贝与浅拷贝如何区分的详细内容,更多请关注创想鸟其它相关文章!
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1408226.html
微信扫一扫
支付宝扫一扫