一尘不染

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

go

如果我将类型定义为:

type T struct {
    S  string
    is []int
}

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

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

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

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


阅读 185

收藏
2020-07-02

共1个答案

一尘不染

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

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

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

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

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

如果您拥有(或可以修改)包,则最好是提供一个Clone()方法或函数,或者为SetIs()type
提供一个方法T。如果第三方软件包不提供此类功能,则您无能为力。

请注意,使用包unsafe可以执行此类操作,但顾名思义,它是 不安全的
,您应远离它。

另外请注意,您可以创建新的价值T在那里is是不可复制的,但将是零值,它的类型(这在以下情况下[]intnil):

var r somepackage.T
s := somepackage.T{S: p.S}

fmt.Printf("%q\n", r)
fmt.Printf("%q\n", s)

将输出:

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

但是您不能为unexported字段设置任何非零值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)

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

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

我们甚至可以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)

Go Playground上尝试一下。

但是我们不能做的
就是将person.age未导出的字段更改为指向其他内容。没有声明包的帮助,它只能是nil或相同的指针值(指向对象的原始字段)。

2020-07-02