你不能 这就是未导出字段的意义:只有声明包才能修改它们。
请注意,如果
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是不可复制的,但将是零值,它的类型(这在以下情况下
[]int会
nil):
var r somepackage.Ts := somepackage.T{S: p.S}fmt.Printf("%qn", r)fmt.Printf("%qn", s)将输出:
{"" []}{"some string" []}但是您不能为unexported字段设置任何非零值
T.is。
请注意,您可以通过简单地将它们分配给另一个结构变量(相同类型)来“精确”复制具有未导出字段的结构,这也将正确地复制未导出字段。
像这个例子一样:
type person struct { Name string age *int}age := 22p := &person{"Bob", &age}fmt.Println(p)p2 := new(person)*p2 = *pfmt.Println(p2)将输出(在Go Playground上尝试):
&{Bob 0x414020}&{Bob 0x414020}我们甚至可以
reflect在不依赖具体类型的情况下进行概括:
type person struct { Name string age *int}age := 22p := &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或相同的指针值(指向对象的原始字段)。



