如何克隆具有未导出字段的结构?
How to clone a structure with unexported field?
如果我将类型定义为:
type T struct {
S string
is []int
}
那我该如何克隆这种类型的对象呢?如果我做一个简单的作业:
p := T{"some string", []int{10, 20}}
q := p
然后对 []int
所做的任何更改都会影响这两个对象。由于 T.is
未导出,因此无法显式复制,即使使用反射提取也是如此。
我目前在类型本身的包中提供了一个 Clone
方法。但这对其他包中的类似类型没有帮助。还有其他方法吗?
你不能。这就是未导出字段的意义所在:只有声明包才能修改它们。
注意,如果T
类型声明在另一个包中,你甚至不能写:
p := somepackage.T{"some string", []int{10, 20}}
因为这会隐式尝试设置未导出的 T.is
字段,从而导致编译时错误:
implicit assignment of unexported field 'is' in somepackage.T literal
如果您拥有(或可以修改)包,最好提供一个 Clone()
方法或函数,或者为类型 T
提供一个 SetIs()
方法。如果第 3 方包不提供此类功能,您将无能为力。
注意使用包 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
。
请注意,您可以通过简单地 assigning 将具有未导出字段的结构复制 "exact" 到另一个结构变量(相同类型),这也将正确复制未导出字段。
就像这个例子:
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
或相同的指针值(指向对象作为原始字段)。
如果我将类型定义为:
type T struct {
S string
is []int
}
那我该如何克隆这种类型的对象呢?如果我做一个简单的作业:
p := T{"some string", []int{10, 20}}
q := p
然后对 []int
所做的任何更改都会影响这两个对象。由于 T.is
未导出,因此无法显式复制,即使使用反射提取也是如此。
我目前在类型本身的包中提供了一个 Clone
方法。但这对其他包中的类似类型没有帮助。还有其他方法吗?
你不能。这就是未导出字段的意义所在:只有声明包才能修改它们。
注意,如果T
类型声明在另一个包中,你甚至不能写:
p := somepackage.T{"some string", []int{10, 20}}
因为这会隐式尝试设置未导出的 T.is
字段,从而导致编译时错误:
implicit assignment of unexported field 'is' in somepackage.T literal
如果您拥有(或可以修改)包,最好提供一个 Clone()
方法或函数,或者为类型 T
提供一个 SetIs()
方法。如果第 3 方包不提供此类功能,您将无能为力。
注意使用包 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
。
请注意,您可以通过简单地 assigning 将具有未导出字段的结构复制 "exact" 到另一个结构变量(相同类型),这也将正确复制未导出字段。
就像这个例子:
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
或相同的指针值(指向对象作为原始字段)。