登录
首页 >  Golang >  Go问答

如何在 Go 中复制接口值?

来源:Golang技术栈

时间:2023-04-18 12:09:07 443浏览 收藏

Golang小白一枚,正在不断学习积累知识,现将学习到的知识记录一下,也是将我的所得分享给大家!而今天这篇文章《如何在 Go 中复制接口值?》带大家来了解一下如何在 Go 中复制接口值?,希望对大家的知识积累有所帮助,从而弥补自己的不足,助力实战开发!


问题内容

如何在 Go 中复制接口值?

我的User界面:

type User interface {
    Name() string
    SetName(name string)
}

我的Admin结构:

type Admin struct {
    name string
}

func (a *Admin) Name() string {
    return a.name
}

func (a *Admin) SetName(name string) {
    a.name = name
}

我尝试复制user1的价值。

主功能:

func main() {
    var user1 User
    user1 = &Admin{name:"user1"}

    fmt.Printf("User1's name: %s\n", user1.Name())

    var user2 User
    user2 = user1
    user2.SetName("user2")

    fmt.Printf("User2's name: %s\n", user2.Name()) // The name will be changed as "user2"
    fmt.Printf("User1's name: %s\n", user1.Name())  // The name will be changed as "user2" too, How to make the user1 name does not change锛�
}

如何实现更改副本名称而不更改原始名称?

正确答案

这里的问题是您的user1变量(类型为User)包含一个 指向 结构的指针Admin

当您分配user1给另一个变量(类型User)时,将复制作为动态类型和值对的接口值(value;type)- 因此将复制指向同一Admin结构的指针。所以你只有一个Admin结构值,两者都user1引用user2(指向)这个。通过任何接口值更改它会更改唯一值。

要制作user1user2区分,您需要 2 个“基础”Admin结构。

一种方法是在接口值中键入断言user1值,并制作该结构的副本,并将其地址包装在另一个User值中:

var user2 User
padmin := user1.(*Admin) // Obtain *Admin pointer
admin2 := *padmin        // Make a copy of the Admin struct
user2 = &admin2          // Wrap its address in another User
user2.SetName("user2")

现在它们将是不同的,输出(在Go Playground上尝试):

User1's name: user1
User2's name: user2
User1's name: user1

当然这个解决方案有它的局限性:存储在接口值中的动态类型在解决方案( )User中是“连线的” 。*Admin

使用反射

如果我们想要一个“通用”的解决方案(不仅仅是一个可以使用的解决方案*Admin),我们可以使用反射(reflect包)。

为简单起见,我们假设user1总是包含一个指针(现在)。

使用反射我们可以得到动态类型(这里*Admin),甚至是没有指针的动态类型(Admin)。我们可以使用reflect.New()来获取指向该类型的新值的指针(其类型将与

  • 中的原始动态类型相同user1*Admin,并将其包装回User. 这就是它的样子:

    var user3 User user3 = reflect.New(reflect.ValueOf(user1).Elem().Type()).Interface().(User) user3.SetName("user3")

输出(在Go Playground上试试这个):

User1's name: user1
User3's name: user3
User1's name: user1

请注意,这reflect.New()将创建一个初始化为零值的新值(因此它不会是原始值的副本)。这不是问题,因为Admin只有一个领域我们将要改变,但总的来说必须牢记在心。

我们最初的假设是user1包含一个指针。现在,“完整”解决方案绝不能做出这样的假设。如果 in 中的值user1不是指针,这就是它可以被“克隆”的方式:

var user3 User
if reflect.TypeOf(user1).Kind() == reflect.Ptr {
    // Pointer:
    user3 = reflect.New(reflect.ValueOf(user1).Elem().Type()).Interface().(User)
} else {
    // Not pointer:
    user3 = reflect.New(reflect.TypeOf(user1)).Elem().Interface().(User)
}
user3.SetName("user3")

文中关于golang的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《如何在 Go 中复制接口值?》文章吧,也可关注golang学习网公众号了解相关技术文章。

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