登录
首页 >  Golang >  Go教程

Golang结构体复制:深拷贝与浅拷贝解析

时间:2026-01-21 11:40:35 357浏览 收藏

有志者,事竟成!如果你在学习Golang,那么本文《Golang结构体复制方法:深拷贝与浅拷贝详解》,就很适合你!文章讲解的知识点主要包括,若是你对本文感兴趣,或者是想搞懂其中某个知识点,就请你继续往下看吧~

Go中结构体默认浅拷贝,赋值时基本类型字段独立,但slice、map、指针、chan、func和interface{}字段共享底层数据;深拷贝需手动实现Clone方法或谨慎使用序列化。

如何在Golang中复制结构体避免指针共享_Golang深拷贝与浅拷贝说明

Go 中结构体默认是浅拷贝,赋值即复制值

Go 的结构体是值类型,直接用 = 赋值时,会逐字段复制。如果字段全是基本类型(intstringbool)或数组,那就真的完全独立;但只要包含指针、slicemapchaninterface{},这些字段的底层数据仍被共享。

哪些字段会导致浅拷贝后修改影响原结构体

以下字段类型在复制后仍与原结构体共用底层数据:

  • slice:新结构体的 slice 与原结构体指向同一底层数组
  • map:两个结构体的 map 变量引用同一个哈希表
  • *T(任意指针):复制的是地址值,指向同一块内存
  • chanfunc 类型同理,复制的是引用

例如:

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/gobencoding/json 序列化再反序列化(注意:要求字段可导出,且 json 不支持 funcchan、循环引用)
  • 第三方库如 github.com/jinzhu/copiergithub.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
}

为什么不要盲目用 JSON 做深拷贝

看似方便,但有多个隐性陷阱:

  • time.Time 会被转成字符串再解析,丢失精度或时区信息(取决于 JSONTime 设置)
  • 未导出字段(小写首字母)完全被忽略,静默丢弃
  • nil slice 和空 slice 在 JSON 中都变成 [],无法区分
  • 嵌套结构体中含 funcunsafe.Pointersync.Mutex 等会 panic
  • 性能差:涉及内存分配、反射、编码/解码,比手写复制慢一个数量级以上

仅适合临时调试或原型阶段,生产代码慎用。

深拷贝的关键不是“有没有工具”,而是清楚每个字段的语义——是该共享还是隔离。结构体设计时就该决定哪些字段用值类型、哪些该封装成带 Copy() 方法的自定义类型,比事后补深拷贝更可靠。

以上就是《Golang结构体复制:深拷贝与浅拷贝解析》的详细内容,更多关于的资料请关注golang学习网公众号!

前往漫画官网入口并下载 ➜
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>