一分耕耘,一分收获!既然都打开这篇《具有结构体的结构体指针的接口的函数赋值显示不同的值》,就坚持看下去,学下去吧!本文主要会给大家讲到等等知识点,如果大家对本文有好的建议或者看到有不足之处,非常欢迎大家积极提出!在后续文章我会继续更新Golang相关的内容,希望对大家都有所帮助!
问题内容我对代码的下一部分有点困惑:
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.weight
和 g := 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
是嵌入式 tiger
类型的升级方法,因此 t.weightzqbendczq b 实际上意味着/表示
asiantiger.tiger.weight( )
方法,其接收者是 tiger
结构体值。)
另一方面,方法值i.weight
是接口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
的实际值(以及指向 t
的 i
)。
剩余的代码:
z := t
x := i
t.weight = 8
fmt.Println(z.Weight(), x.Weight()) // result is: 8 8
这里 z
和 x
不是方法值。 z
是 t
的副本(指针),而 x
是 i
(包装指针的接口值)的副本。并且您更改指向值(指向值的字段的字段)。最后调用 z
和 x
的方法,不会发生保存的接收器,接收器在更改后进行评估。
理论要掌握,实操不能落!以上关于《具有结构体的结构体指针的接口的函数赋值显示不同的值》的详细介绍,大家都掌握了吧!如果想要继续提升自己的能力,那么就来关注编程网公众号吧!