首頁 > 軟體

golang值接收者和指標接收者的區別介紹

2022-08-29 22:00:18

方法

方法能給使用者自定義的型別新增新的行為。它和函數的區別在於方法有一個接收者,給一個函數新增一個接收者,那麼它就變成了方法。接收者可以是值接收者,也可以是指標接收者
在呼叫方法的時候,值型別既可以呼叫值接收者的方法,也可以呼叫指標接收者的方法;指標型別既可以呼叫指標接收者的方法,也可以呼叫值接收者的方法。

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 的所有方法

兩者分別在何時使用

如果方法的接收者是值型別,無論呼叫者是物件還是物件指標,修改的都是物件的副本,不影響呼叫者;如果方法的接收者是指標型別,則呼叫者修改的是指標指向的物件本身

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

  • 方法能夠修改接收者指向的值。
  • 避免在每次呼叫方法時複製該值,在值的型別為大型結構體時,這樣做會更加高效。
  • 是使用值接收者還是指標接收者,不是由該方法是否修改了呼叫者(也就是接收者)來決定,而是應該基於該型別的本質。

到此這篇關於golang值接收者和指標接收者的區別的文章就介紹到這了,更多相關golang值接收者和指標接收者內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com