Go 中接收器方法调用语法混乱
在探索接收器的指针和值之间的差异时,请注意,值方法可以是对指针和值都调用,而指针方法仅限于指针。此限制源于指针方法修改接收器的能力,如果在值的副本上调用,则会导致丢弃修改。
但是,显示对值调用指针方法的代码示例会导致混乱。代码如下:
<code class="go">package main import ( "fmt" "reflect" ) type age int func (a age) String() string { return fmt.Sprintf("%d year(s) old", int(a)) } func (a *age) Set(newAge int) { if newAge >= 0 { *a = age(newAge) } } func main() { var vAge age = 5 pAge := new(age) fmt.Printf("Type information:\n\tvAge: %v\n\tpAge: %v\n", reflect.TypeOf(vAge), reflect.TypeOf(pAge)) fmt.Printf("vAge.String(): %v\n", vAge.String()) fmt.Printf("vAge.Set(10)\n") vAge.Set(10) fmt.Printf("vAge.String(): %v\n", vAge.String()) fmt.Printf("pAge.String(): %v\n", pAge.String()) fmt.Printf("pAge.Set(10)\n") pAge.Set(10) fmt.Printf("pAge.String(): %v\n", pAge.String()) }</code>
尽管文档说明这是不可能的,但是这段代码编译并运行没有错误。那么,是什么导致了这种明显的不一致呢?
答案在于可寻址值的概念中。 Go 规范定义“如果 x(的类型)的方法集包含 m 并且参数列表可以分配给 m 的参数列表,则方法调用 x.m() 是有效的”。此外,“如果 x 是可寻址的,并且 &x 的方法集包含 m,则 x.m() 是 (&x).m() 的简写。”
在这种情况下,vAge 是可寻址的,这意味着它具有有效的内存地址。当调用 vAge.Set() 时,编译器识别出指针接收器方法 Set() 也可以使用值的地址来调用,相当于 (&vAge).Set().
以上是## 为什么我可以在 Go 中调用值的指针方法?的详细内容。更多信息请关注PHP中文网其他相关文章!