登录
首页 >  Golang >  Go问答

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

来源:Golang技术栈

时间:2023-03-08 07:50:30 104浏览 收藏

有志者,事竟成!如果你在学习Golang,那么本文《如何克隆具有未导出字段的结构?》,就很适合你!文章讲解的知识点主要包括golang,若是你对本文感兴趣,或者是想搞懂其中某个知识点,就请你继续往下看吧~

问题内容

如果我有一个类型定义为:

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()type 提供一个方法T。如果第 3 方软件包不提供此类功能,则您无能为力。

请注意,使用 packageunsafe可以做这样的事情,但正如它的名字所说:它是 不安全 的,你应该远离它。

另请注意,您可以创建未复制Twhere的新值,但将是其类型的零值(在will be的情况下):is[]int``nil

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

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

这将输出:

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

但是您不能为 unexported field 设置任何非零值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或相同的指针值(指向对象作为原始字段)。

今天带大家了解了golang的相关知识,希望对你有所帮助;关于Golang的技术知识我们会一点点深入介绍,欢迎大家关注golang学习网公众号,一起学习编程~

声明:本文转载于:Golang技术栈 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>
评论列表