一分耕耘,一分收获!既然都打开这篇《具有结构体的结构体指针的接口的函数赋值显示不同的值》,就坚持看下去,学下去吧!本文主要会给大家讲到等等知识点,如果大家对本文有好的建议或者看到有不足之处,非常欢迎大家积极提出!在后续文章我会继续更新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 的方法,不会发生保存的接收器,接收器在更改后进行评估。
理论要掌握,实操不能落!以上关于《具有结构体的结构体指针的接口的函数赋值显示不同的值》的详细介绍,大家都掌握了吧!