Go には、Java などのオブジェクト指向言語で見られるメソッドのオーバーライドに直接相当するものはありません。ただし、同様の機能を実現する別のアプローチもあります。
文字列 "base" を返すメソッド Get() を持つ基本型 Base があるシナリオを考えてみましょう。 Base から継承する新しい型 Sub を定義したいとしますが、「Sub」を返す Get() の実装が異なります。
次のような合成を使用する最初の試み:
type Sub struct { Base } func(sub *Sub) Get() string { return "Sub" }
Sub の Get() メソッドは Base の Get() メソッドをオーバーライドできないため、機能しません。これは、Go のメソッド レシーバー セマンティクスによるもので、レシーバーのタイプ (この場合は *Base) によって呼び出されるメソッドが決まります。
1 つの代替アプローチは、インターフェイスを使用することです。インターフェイスは、Java の抽象クラスと同様に、型が実装する必要があるメソッドのコントラクトを定義します。この場合、単一の Get() メソッドを使用して Getter インターフェイスを定義できます。
type Getter interface { Get() string }
その後、Base を変更して Getter インターフェイスを実装できます。
type Base struct { } func (base *Base) Get() string { return "base" }
ここで、 Base のインスタンスを埋め込み、Getter も実装する構造体として Sub を定義できます:
type Sub struct { Base } func (sub *Sub) Get() string { return "Sub" }
GetName() メソッドを使用して、呼び出す Get() の実装を動的に決定します。
func (base *Base) GetName(getter Getter) string { if g, ok := getter.(Getter); ok { return g.Get() } else { return base.Get() } }
main 関数で、Sub のインスタンスを作成し、それを Getter にキャストし、それを GetName( ):
userType := Sub{} fmt.Println(userType.GetName()) // prints "Sub"
もう 1 つの方法は、メソッドの埋め込みを使用することです。これにより、メソッドを実装する型を埋め込むことで、メソッドを構造体に追加できます。この場合、Base を埋め込んで新しい Get() メソッドを実装する新しいタイプの SubGetter を定義できます:
type SubGetter struct { Base } func (sub *SubGetter) Get() string { return "Sub" }
その後、SubGetter を Base のメソッドとして使用できます:
type Base struct { SubGetter }
これにより、Base:
base := Base{} fmt.Println(base.Get()) // prints "Sub"のメソッドであるかのように、SubGetter の Get() メソッドを呼び出すことができます。
以上がGo でメソッドオーバーライド機能を実現するにはどうすればよいですか?の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。