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中文網其他相關文章!