網站首頁 編程語言 正文
方法
方法能給用戶自定義的類型添加新的行為。它和函數的區別在于方法有一個接收者,給一個函數添加一個接收者,那么它就變成了方法。接收者可以是值接收者,也可以是指針接收者。
在調用方法的時候,值類型既可以調用值接收者的方法,也可以調用指針接收者的方法;指針類型既可以調用指針接收者的方法,也可以調用值接收者的方法。
package main import "fmt" type Person struct { age int } func (p Person) AddAge() { p.age += 1 } func (p *Person) GetAge() { p.age += 1 } func main() { // p1 是值類型 p := Person{age: 18} // 值類型 調用接收者也是值類型的方法 p.AddAge() fmt.Println(p.age) // ---------------------- // p2 是指針類型 指針類型調用接收者是值類型的方法 p2 := &Person{age: 100} p2.AddAge() fmt.Println(p2.age) //值類型 調用接收者也是指針類型的方法 p3 := Person{age: 18} p3.GetAge() fmt.Println(p3.age) // 指針類型 調用收者也是指針類型的方法 p4 := Person{age: 100} p4.GetAge() fmt.Println(p4.age) } //18 //100 //19 //101
值接收者 | 指針接收者 | |
---|---|---|
值類型調用者 | 傳遞一個副本 | 使用值的引用來調用方法 |
指針類型調用者 | 傳遞一個副本 | 方法里的操作會影響到調用者,類似于指針傳參,拷貝了一份指針 |
總結:
1.一個結構體的方法的接收者可能是類型值或指針
2.當接受者不是一個指針時,該方法操作對應接受者的值的副本,即使你使用了指針調用函數,但是函數的接受者是值類型,所以函數內部操作還是對副本的操作,而不是指針操作。
3.如果接收者是指針,則調用者修改的是指針指向的值本身。
接口實現
當結構體實現一個接口時,可以在方法中設置一個接收者,比如對于以下接口:
type Inter interface { foo() }
結構體在實現它時,方法的接收者類型可以是:值、指針。比如:
type S struct {} func (s S) foo() {} // 值類型 func (s *S) foo() {} // 或者指針類型
在使用結構體初始化接口變量時,結構體的類型也可以是:值、指針。比如:
//賦值 var s Inter = S{} // 值類型 s.foo() var s Inter = &S{} // 指針類型 s.foo()
那么調用接口方法的組合實際有四種情況:
值類型結構體 -> 賦值給接口 -> 調用接收者類型為值類型的結構體方法
指針類型結構體 -> 賦值給接口 -> 調用接收者類型為指針類型的結構體方法
值類型結構體 -> 賦值給接口 -> 調用接收者類型為指針類型的結構體方法(不通過)
指針類型結構體 -> 賦值給接口 -> 調用接收者類型為值類型的結構體方法
結構體類型為值類型、調用了接收者為指針的方法不通過。但是反過來,結構體為指針類型時,卻可以調用接收值為值或指針的任何方法。這是為什么呢?
接收者是方法的一個額外的參數,而 Go 在調用函數的時候,參數都是值傳遞的。將一個指針拷貝,它們還是指向同一個地址,指向一個確定的結構體;將一個值拷貝,它們變成了兩個不同的結構體,有著不同的地址。這會導致以下兩種情況:
當在一個結構體指針上,通過接口,調用一個接收者為值類型的方法時,Go 首先會創建這個指針的副本,然后將這個指針解引用,再作為接收者參數傳遞給該方法。這兩個指針指向相同的地址,所以它們傳遞給方法的接收者參數都是相同的。
type Inter interface { foo() } type S struct {} func (s S) foo() {} // 接收者為值類型的方法 var a Inter = &S{} // 使用結構體指針初始化一個接口 a.foo() // 調用 foo 方法 // 實際上底層是這樣的: // 首先拷貝 a 的底層值,即 `&S{}`,是一個結構體指針: var b *S = a.inner_value // a、b 是不同的變量,但是指向同一個結構體 // 然后將 b 解引用,傳遞給 foo: foo(*b) // *b 和 *(a.inner_value) 其實都表示同一個結構體
這些規則用來說明是否我們一個類型的值或者指針實現了該接口:
- 類型 *T 的可調用方法集包含接受者為 *T 或 T 的所有方法集
- 類型 T 的可調用方法集包含接受者為 T 的所有方法
兩者分別在何時使用
如果方法的接收者是值類型,無論調用者是對象還是對象指針,修改的都是對象的副本,不影響調用者;如果方法的接收者是指針類型,則調用者修改的是指針指向的對象本身。
使用指針作為方法的接收者的理由:
- 方法能夠修改接收者指向的值。
- 避免在每次調用方法時復制該值,在值的類型為大型結構體時,這樣做會更加高效。
- 是使用值接收者還是指針接收者,不是由該方法是否修改了調用者(也就是接收者)來決定,而是應該基于該類型的本質。
原文鏈接:https://blog.csdn.net/weixin_42128977/article/details/126511071
相關推薦
- 2022-02-26 C#中XML基礎用法_C#教程
- 2022-09-23 Python文件目錄操作常用模塊的使用詳解_python
- 2022-03-16 C++?explicit關鍵字講解_C 語言
- 2022-07-26 分析查詢語句 Explain
- 2022-04-23 Android如何使用ViewPager2實現頁面滑動切換效果_Android
- 2022-02-12 使用background-attachment實現視差滾動、水波
- 2023-03-16 Python?asyncio異步編程簡單實現示例_python
- 2023-07-16 springboot動態端口
- 最近更新
-
- 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同步修改后的遠程分支