如何使用未导出的字段克隆结构?

时间:2016-07-14 08:37:42

标签: go struct clone

如果我的类型定义为:

type T struct {
    S  string
    is []int
}

然后我该如何克隆这种类型的对象?如果我做了一个简单的任务:

p := T{"some string", []int{10, 20}}
q := p

然后对[]int所做的任何更改都会影响这两个对象。由于未导出T.is,因此无法显式复制,即使使用反射提取也是如此。

我目前正在该类型的包中提供Clone方法。但这对其他软件包中的类似类型没有帮助。还有另一种方法吗?

1 个答案:

答案 0 :(得分:6)

你不能。这是未导出字段的重点:只有声明包才能修改它们。

请注意,如果T类型在另一个包中声明,您甚至无法写:

p := somepackage.T{"some string", []int{10, 20}}

因为这会隐式尝试设置未导出的T.is字段,从而导致编译时错误:

implicit assignment of unexported field 'is' in somepackage.T literal

如果您拥有(或者您可以修改)包,最好是提供Clone()方法或功能,或者为SetIs()类型提供T方法。如果第三方软件包没有提供此类功能,那么您无能为力。

请注意,使用包unsafe可以执行此类操作,但正如其名称所示:它不安全,您应该远离它。功能

另请注意,您可以创建T的新值,其中is未被复制,但将是其类型的zero value(如果[]int将是nil var r somepackage.T s := somepackage.T{S: p.S} fmt.Printf("%q\n", r) fmt.Printf("%q\n", s) ):

{"" []}
{"some string" []}

将输出:

T.is

但是,您无法为未导出的字段type person struct { Name string age *int } age := 22 p := &person{"Bob", &age} fmt.Println(p) p2 := new(person) *p2 = *p fmt.Println(p2) 设置任何非零值。

请注意,您可以制作"确切"具有未导出字段的结构的副本只需assigning到另一个结构变量(相同类型),这也将正确复制未导出的字段。

就像这个例子一样:

&{Bob 0x414020}
&{Bob 0x414020}

将输出(在Go Playground上尝试):

reflect

我们甚至可以使用type person struct { Name string age *int } age := 22 p := &person{"Bob", &age} fmt.Println(p) v := reflect.ValueOf(p).Elem() vp2 := reflect.New(v.Type()) vp2.Elem().Set(v) fmt.Println(vp2) 来概括而不依赖于具体类型:

person.age

Go Playground上试试这个。

但我们无法做的是将nil未导出的字段更改为指向其他字段。在没有声明包的帮助下,它只能是myFunction(\'$([[#new-'或相同的指针值(指向对象作为原始字段)。