網站首頁 編程語言 正文
引言
在review 一些代碼中,發現經常某個類型定義的方法,其接收者既有值類型,又有指針類型,然后 Goland 就有提示: Struct Person has methods on both value and pointer receivers. Such usage is not recommended by the Go Documentation.
一般來講,這個提示對代碼的運行并不會產生什么問題。只不過對于有輕微 “代碼潔癖” 的人來講,體感不好,就一定想要改統一。
當然,我并不是想講要統一的問題,前面說這么多廢話,只是為了鋪墊一下引出本文的內容:Go中的值接收者與指針接收者有什么關系與區別,該怎么選?
聯系與區別
在繼續講下去之前,我們得先明確,Go 里邊能夠定義方法的必須是自定義類型,而不能是系統內置類型,比如 int、string 這種是不可以為其添加方法的。
那么當我們定義了一個自定義類型,可以為其添加方法,先上代碼:
package main import "fmt" type Person struct { name string age int } // 值針接收者 func (p Person) GetName() string { return p.name } // 指針接收者 func (p *Person) GetAge() int { return p.age } func main() { // 定義了一個【值類型】 t := Person{ name: "DaYu", age: int(28), } // 調用值方法 fmt.Println(t.GetName()) // 調用指針方法 fmt.Println(t.GetAge()) } -----運行結果------- study/demo01/client go run * DaYu 28
指針類型調用結果
從使用過程看,值類型的變量,可以調用該類型的值接收者方法,也可以調用指針接收者方法。
反之,我們可以定義一個指針類型,然后看看調用結果:
package main import "fmt" type Person struct { name string age int } func (p *Person) GetName() string { return p.name } func (p Person) GetAge() int { return p.age } func main() { // 注意,其它地方都沒有改,只是這里改變了類型 t := &Person{ name: "DaYu", age: int(28), } fmt.Println(t.GetName()) fmt.Println(t.GetAge()) } -----運行結果------- study/demo01/client go run * DaYu 28
這段代碼告訴我們,指針類型的變量,可以調用該類型的值接收者方法,也可以調用指針接收者方法。
是不是特別有意思?
- 值類型變量,可以調用值接收的方法,也可以調用指針接收者的方法;
- 指針類型變量,可以調用值接收的方法,也可以調用指針接收者的方法。
看起來好像兩者對等的,并沒有差別。那么二者真的沒有差別嗎?只是一種表達形式上的差異?其實不然,如果引入接口類型后,我們再來看看。
package main // 新增的接口 type Animal interface { GetName() string GetAge() int } type Person struct { name string age int } func (p *Person) GetName() string { return p.name } func (p Person) GetAge() int { return p.age } func main() { // 定義的接口變量 var ani Animal // person 實現了 Animal 接口,賦值給了 ani 變量 // 但是,這里編譯會通不過,錯誤如下: // Cannot use 'Person{ name: "DaYu", age: int(28), }' (type Person) as the type Animal Type does not implement 'Animal' as the 'GetName' method has a pointer receiver ani = Person{ name: "DaYu", age: int(28), } ani.GetName() ani.GetAge() }
為什么會報錯呢? 錯誤提醒很明顯了:Person 沒有實現 Animal 的 GetName 方法。因為在上面的代碼中,我們實現 GetName 方法的是 (*Person) 類型。
但是為什么 GetAge 方法不報錯呢? 那是因為 Go 里邊對于 (Type)Method 的方法,會自動讓他擁有 (*Type)Method 方法的能力。
實現接口時約束
- 如果定義的是 (Type)Method,則該類型會隱式的聲明一個 (*Type)Method;
- 如果定義的是 (*Type)Method ,則不會隱式什么一個 (Type)Method。
至于為什么不也隱式申明一個 (Type)Method ,我覺得有一個原因是,我們一般采用指針接收者時,方法內部改變的值,接收者本身也會改變,那么此時如果隱式有這樣一個申明,外部使用值類型時,這個改變就不會生效,語義上就會非常奇怪。
該怎么用
從使用表現上看,指針接收者在方法內部的改變,會體現到其本身。但這并不是決定我們要不要用指針接收者的唯一理由! 最重要的還是看接收者要不要全局共享一個實體,其次某些場景下,如果接收者本身太大,拷貝成本很高,也應該使用指針接收者。
回到文檔開篇的問題,為什么不建議值接收者、指針接收者混用,主要還是在于語義不夠清晰,存在潛在理解成本的問題。
原文鏈接:https://juejin.cn/post/7158354728961703973
相關推薦
- 2023-02-14 Cython處理C字符串的示例詳解_python
- 2022-06-19 python繪制散點圖和折線圖的方法_python
- 2021-11-09 C++11?thread多線程編程創建方式_C 語言
- 2022-10-29 Spring的純注解配置詳解
- 2022-05-06 Python判斷字符串中是否是中英文文小技巧
- 2022-09-17 Pandas中df.loc[]與df.iloc[]的用法與異同?_python
- 2022-10-13 python中arrow庫用法大全_python
- 2022-05-02 Redis使用命令行與多數據庫配置_Redis
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支