1 回答

TA貢獻(xiàn)1871條經(jīng)驗(yàn) 獲得超13個(gè)贊
func (s *MyStruct) pointerMethod() { } // method on pointer
func (s MyStruct) valueMethod() { } // method on value
對(duì)于不習(xí)慣指針的程序員來說,這兩個(gè)例子之間的區(qū)別可能會(huì)令人困惑,但情況其實(shí)很簡(jiǎn)單。在類型上定義方法時(shí),接收器(在上面的示例中為 s)的行為就像它是方法的參數(shù)一樣。將接收者定義為值還是指針是同一個(gè)問題,然后,函數(shù)參數(shù)應(yīng)該是值還是指針。有幾個(gè)考慮。
首先,也是最重要的,該方法是否需要修改接收器?如果是,則接收者必須是一個(gè)指針。(切片和映射充當(dāng)引用,所以它們的故事有點(diǎn)微妙,但是例如要更改方法中切片的長度,接收者仍然必須是指針。)在上面的示例中,如果 pointerMethod 修改了s,調(diào)用者將看到這些更改,但 valueMethod 是使用調(diào)用者參數(shù)的副本(這是傳遞值的定義)調(diào)用的,因此調(diào)用者將看不到它所做的更改。
順便說一句,指針接收器與 Java 中的情況相同,盡管在 Java 中指針隱藏在幕后;Go 的價(jià)值接收者是不尋常的。
二是對(duì)效率的考慮。如果接收器很大,例如一個(gè)大的結(jié)構(gòu),使用指針接收器會(huì)便宜得多。
接下來是一致性。如果該類型的某些方法必須有指針接收器,其余的也應(yīng)該如此,因此無論如何使用該類型,方法集都是一致的。有關(guān)詳細(xì)信息,請(qǐng)參閱方法集部分。
對(duì)于諸如基本類型、切片和小型結(jié)構(gòu)之類的類型,值接收器非常便宜,因此除非方法的語義需要指針,否則值接收器是高效且清晰的。
在 Go 中,所有參數(shù)和返回值都是按值傳遞的。接收器按值傳遞。使用指針接收器更改值。例如,
package main
import (
"fmt"
)
type Prompter interface {
Define(f *Field)
}
type Field struct {
Key string
}
type Provider interface {
Prompt(Prompter)
}
var providers = []Provider{
MyProvider{},
}
type MyProvider struct{}
func (p MyProvider) Prompt(prompter Prompter) {
prompter.Define(&Field{"name"})
prompter.Define(&Field{"age"})
}
type CliPrompter struct {
fields []*Field
}
func NewCliPrompter() *CliPrompter {
return &CliPrompter{
fields: make([]*Field, 0, 100),
}
}
func (c *CliPrompter) Define(f *Field) {
fmt.Printf("fields: %+v\n", c.fields)
c.fields = append(c.fields, f)
fmt.Printf("fields: %+v\n", c.fields)
}
func main() {
providers[0].Prompt(NewCliPrompter())
}
輸出:
fields: []
fields: [0x1040a120]
fields: [0x1040a120]
fields: [0x1040a120 0x1040a130]
- 1 回答
- 0 關(guān)注
- 231 瀏覽