go - 如何克隆具有未导出字段的结构?

标签 go struct clone

如果我将类型定义为:

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() 方法或函数,或者提供一个 SetIs() 方法输入 T。如果第 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 制作具有未导出字段的结构的“精确”副本。它们到另一个结构变量(相同类型),这也将正确复制未导出的字段。

就像这个例子:

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或相同的指针值(指向对象作为原始字段)。

关于go - 如何克隆具有未导出字段的结构?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38369350/

相关文章:

c - 从内核模块中的 AF_UNIX 套接字的 fd 获取绝对路径

flash - 在 ActionScript 3 中克隆影片剪辑

git - 克隆远程 Git 存储库(服务器拒绝您更改到给定目录)

带有自身类类型的 TypeScript 克隆自身实例

go - 正则表达式匹配字符串值并替换 golang 中的所有出现

go - runtime._ExternalCode Cpu 使用率过高,高达 80%

go - 在 Go 模板范围循环中,是否在每次迭代时重置循环外声明的变量?

unit-testing - 如何测试CLI标志-当前因“标志重新定义”而失败

我可以在 c 中指定类似 Java 的 'constructor' 吗?

c++ - 将数组传递给函数