如何复制struct和dereference所有指针

时间:2018-05-10 09:10:37

标签: pointers go struct copy dereference

如何复制Item结构和所有指向新结构的指针?

type Item struct {
    A    []*ASet     `json:"a,omitempty"`
    B    []*BSet.    `json:"b,omitempty"`
    C    []*CSet.    `json:"c,omitempty"`
}


type ASet struct {
    UID   string     `json:"uid,omitempty"`
    Items []*ItemA `json:"member,omitempty"`
}

type ItemA struct {
    UID  string `json:"uid,omitempty"`
    Portset []*PortSet `json:"portset,omitempty"`
}

type PortSet struct {
    UID   string     `json:"uid,omitempty"`
    Ports []*Port `json:"member,omitempty"`
}

type Port struct {
    UID  string `json:"uid,omitempty"`
    Port int    `json:"port,omitempty"`
}

我不希望新结构引用旧结构。

1 个答案:

答案 0 :(得分:2)

您想要的本质上是标准库不支持的深层副本。

您的选择:

  • 复制"手动",例如创建一个新结构并复制字段,其中指针或切片/贴图/通道/等必须以递归方式手动复制。
    这是最简单的方法,将您的结构分配给另一个复制所有字段的结构,因此您基本上只需要培养指针/贴图/切片等(但递归)。
  • 使用外部库,例如github.com/mohae/deepcopygithub.com/ulule/deepcopiergithub.com/mitchellh/copystructure
  • 将您的结构元素化为某种格式(例如JSON),然后将其解组为另一个变量。

最后一个选项可能如下所示:

var i1 Item
data, err := json.Marshal(i1)
if err != nil {
    panic(err)
}

var i2 Item
if err := json.Unmarshal(data, &i2); err != nil {
    panic(err)
}
// i2 holds a deep copy of i1

请注意,编组/解编不是特别有效,而是简单而紧凑。还要注意,这可能不能很好地处理递归数据结构,甚至可能挂起或恐慌(例如,字段指向包含结构),但处理递归结构可能是所有解决方案的问题。另请注意,这不会克隆未导出的字段。

关于这种编组/解组的好处是你可以轻松地创建一个帮助函数来深层复制"任何"值:

func deepCopy(v interface{}) (interface{}, error) {
    data, err := json.Marshal(v)
    if err != nil {
        return nil, err
    }

    vptr := reflect.New(reflect.TypeOf(v))
    err = json.Unmarshal(data, vptr.Interface())
    if err != nil {
        return nil, err
    }
    return vptr.Elem().Interface(), err
}

测试它:

p1 := image.Point{X: 1, Y: 2}
fmt.Printf("p1 %T %+v\n", p1, p1)

p2, err := deepCopy(p1)
if err != nil {
    panic(err)
}

p1.X = 11
fmt.Printf("p1 %T %+v\n", p1, p1)
fmt.Printf("p2 %T %+v\n", p2, p2)

输出(在Go Playground上尝试):

p1 image.Point (1,2)
p1 image.Point (11,2)
p2 image.Point (1,2)