gpt4 book ai didi

go - 具有结构体的结构体指针的接口(interface)的函数赋值显示不同的值

转载 作者:行者123 更新时间:2023-12-02 18:05:09 26 4
gpt4 key购买 nike

我对代码的下一部分有点困惑:

type Tiger struct {
weight int
}

func (t Tiger) Weight() int {
return t.weight
}

type AsianTiger struct {
Tiger
}

type Tigers interface {
Weight() int
}

func main() {
t := &AsianTiger{Tiger{3}}

var i Tigers = t

f := t.Weight
g := i.Weight

t.weight = 7

fmt.Println(f(), g()) // result is: 3 7 | expected 7 7

z := t
x := i

t.weight = 8

fmt.Println(z.Weight(), x.Weight()) // result is: 8 8

}

有人可以帮助我了解 f := t.Weightg := i.Weight 发生了什么,为什么它们显示的结果与我预期的不同相似之处,以及幕后发生了什么?

最佳答案

问题/困惑的根源在于以下几行:

f := t.Weight
g := i.Weight

这些是method values方法值将接收器保存在结果函数值中!并且您的Tiger.Weight()方法具有值接收器(不是指针接收器):

func (t Tiger) Weight() int {
return t.weight
}

这意味着t.Weight将保存接收者t,它是一个Tiger结构,以及当前的 >weight=3 字段。 (注意:在 f:= t.Weight 中,t 确实是一个 *AsianTiger 值,但是 t.Weight code> 是嵌入式 Tiger 类型的升级方法,因此 t.Weight 实际上表示/表示 AsianTiger.Tiger.Weight() 方法其接收者是一个 Tiger 结构值。)

方法值i.Weight是接口(interface)Tigers的一个方法,其中包装的具体值是*类型AsianTiger,一个指针。因此指针值将被保存。

接下来更改权重:

t.weight = 7

这不会影响f,因为Tiger结构值(接收者)被保存,它是一个副本。然而,这会影响g,因为保存的接收器有Tigers,它保存*AsianTiger,并且您更改了指向的值(而不是指针) .

您可以查看是否添加另一行:

fmt.Println(f(), g())               // result is: 3 7 | expected 7 7
fmt.Println(t.Weight(), i.Weight()) // result is: 7 7

此输出(在 Go Playground 上尝试):

3 7
7 7

第二行打印 7 7,因为没有使用保存的接收器值,而是使用 t 的实际值(以及指向的 i)到t)。

剩余的代码:

z := t
x := i

t.weight = 8

fmt.Println(z.Weight(), x.Weight()) // result is: 8 8

这里的zx不是方法值。 z 是指针 t 的副本,x 是接口(interface)值 i 的副本它包装了一个指针。并且您更改指向值(指向值的字段的字段)。最后调用 zx 的方法,不会保存接收器,而是在更改后评估接收器。

关于go - 具有结构体的结构体指针的接口(interface)的函数赋值显示不同的值,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/73445297/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com