溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

golang值接收者和指針接收者的區(qū)別是什么

發(fā)布時間:2022-08-30 09:31:30 來源:億速云 閱讀:100 作者:iii 欄目:開發(fā)技術(shù)

這篇文章主要講解了“golang值接收者和指針接收者的區(qū)別是什么”,文中的講解內(nèi)容簡單清晰,易于學(xué)習(xí)與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學(xué)習(xí)“golang值接收者和指針接收者的區(qū)別是什么”吧!

方法

方法能給用戶自定義的類型添加新的行為。它和函數(shù)的區(qū)別在于方法有一個接收者,給一個函數(shù)添加一個接收者,那么它就變成了方法。接收者可以是值接收者,也可以是指針接收者。
在調(diào)用方法的時候,值類型既可以調(diào)用值接收者的方法,也可以調(diào)用指針接收者的方法;指針類型既可以調(diào)用指針接收者的方法,也可以調(diào)用值接收者的方法。

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}
	// 值類型 調(diào)用接收者也是值類型的方法
	p.AddAge()
	fmt.Println(p.age)

	// ----------------------

	// p2 是指針類型  指針類型調(diào)用接收者是值類型的方法
	p2 := &Person{age: 100}
	p2.AddAge()
	fmt.Println(p2.age)
	//值類型 調(diào)用接收者也是指針類型的方法
	p3 := Person{age: 18}
	p3.GetAge()
	fmt.Println(p3.age)
	// 指針類型 調(diào)用收者也是指針類型的方法
	p4 := Person{age: 100}
	p4.GetAge()
	fmt.Println(p4.age)
}
//18
//100
//19
//101
值接收者指針接收者
值類型調(diào)用者傳遞一個副本使用值的引用來調(diào)用方法
指針類型調(diào)用者傳遞一個副本方法里的操作會影響到調(diào)用者,類似于指針傳參,拷貝了一份指針

總結(jié):
1.一個結(jié)構(gòu)體的方法的接收者可能是類型值或指針
2.當(dāng)接受者不是一個指針時,該方法操作對應(yīng)接受者的值的副本,即使你使用了指針調(diào)用函數(shù),但是函數(shù)的接受者是值類型,所以函數(shù)內(nèi)部操作還是對副本的操作,而不是指針操作。
3.如果接收者是指針,則調(diào)用者修改的是指針指向的值本身。

接口實現(xiàn)

當(dāng)結(jié)構(gòu)體實現(xiàn)一個接口時,可以在方法中設(shè)置一個接收者,比如對于以下接口:

type Inter interface {
    foo()
}

結(jié)構(gòu)體在實現(xiàn)它時,方法的接收者類型可以是:值、指針。比如:

type S struct {}

func (s S) foo() {} // 值類型
func (s *S) foo() {} // 或者指針類型

在使用結(jié)構(gòu)體初始化接口變量時,結(jié)構(gòu)體的類型也可以是:值、指針。比如:

//賦值
var s Inter = S{} // 值類型
s.foo()

var s Inter = &S{} // 指針類型
s.foo()

那么調(diào)用接口方法的組合實際有四種情況:
值類型結(jié)構(gòu)體 -> 賦值給接口 -> 調(diào)用接收者類型為值類型的結(jié)構(gòu)體方法
指針類型結(jié)構(gòu)體 -> 賦值給接口 -> 調(diào)用接收者類型為指針類型的結(jié)構(gòu)體方法
值類型結(jié)構(gòu)體 -> 賦值給接口 -> 調(diào)用接收者類型為指針類型的結(jié)構(gòu)體方法(不通過)
指針類型結(jié)構(gòu)體 -> 賦值給接口 -> 調(diào)用接收者類型為值類型的結(jié)構(gòu)體方法

結(jié)構(gòu)體類型為值類型、調(diào)用了接收者為指針的方法不通過。但是反過來,結(jié)構(gòu)體為指針類型時,卻可以調(diào)用接收值為值或指針的任何方法。這是為什么呢?

接收者是方法的一個額外的參數(shù),而 Go 在調(diào)用函數(shù)的時候,參數(shù)都是值傳遞的。將一個指針拷貝,它們還是指向同一個地址,指向一個確定的結(jié)構(gòu)體;將一個值拷貝,它們變成了兩個不同的結(jié)構(gòu)體,有著不同的地址。這會導(dǎo)致以下兩種情況:

當(dāng)在一個結(jié)構(gòu)體指針上,通過接口,調(diào)用一個接收者為值類型的方法時,Go 首先會創(chuàng)建這個指針的副本,然后將這個指針解引用,再作為接收者參數(shù)傳遞給該方法。這兩個指針指向相同的地址,所以它們傳遞給方法的接收者參數(shù)都是相同的。

type Inter interface {
    foo()
}
type S struct {}
func (s S) foo() {} // 接收者為值類型的方法

var a Inter = &S{} // 使用結(jié)構(gòu)體指針初始化一個接口
a.foo() // 調(diào)用 foo 方法

// 實際上底層是這樣的:
// 首先拷貝 a 的底層值,即 `&S{}`,是一個結(jié)構(gòu)體指針:
var b *S = a.inner_value // a、b 是不同的變量,但是指向同一個結(jié)構(gòu)體
// 然后將 b 解引用,傳遞給 foo:
foo(*b) // *b 和 *(a.inner_value) 其實都表示同一個結(jié)構(gòu)體

這些規(guī)則用來說明是否我們一個類型的值或者指針實現(xiàn)了該接口:

  • 類型 *T 的可調(diào)用方法集包含接受者為 *T 或 T 的所有方法集

  • 類型 T 的可調(diào)用方法集包含接受者為 T 的所有方法

兩者分別在何時使用

如果方法的接收者是值類型,無論調(diào)用者是對象還是對象指針,修改的都是對象的副本,不影響調(diào)用者;如果方法的接收者是指針類型,則調(diào)用者修改的是指針指向的對象本身。

使用指針作為方法的接收者的理由:

  • 方法能夠修改接收者指向的值。

  • 避免在每次調(diào)用方法時復(fù)制該值,在值的類型為大型結(jié)構(gòu)體時,這樣做會更加高效。

  • 是使用值接收者還是指針接收者,不是由該方法是否修改了調(diào)用者(也就是接收者)來決定,而是應(yīng)該基于該類型的本質(zhì)。

感謝各位的閱讀,以上就是“golang值接收者和指針接收者的區(qū)別是什么”的內(nèi)容了,經(jīng)過本文的學(xué)習(xí)后,相信大家對golang值接收者和指針接收者的區(qū)別是什么這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關(guān)知識點的文章,歡迎關(guān)注!

向AI問一下細節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI