レシーバの違いによる挙動の違い

Ken published on
2 min, 223 words

Categories: Programming

ある構造体のフィールドを得ようと実装したら、なぜか継承元(Goではこう言わないか)のフィールドしか取得できなかった。 構造体を引数に実行すれば取得できるけど、同じメソッドを呼んでいるのだが…。

これはどう理解したらいいのかわからない。

http://play.golang.org/p/KcfBp2myhR

package main

import (
    "fmt"
    "reflect"
)

type I interface {
    Call()
    Method(interface{})
}

type A struct {
    a1 bool
}

func (a *A) Call() {
    a.Method(a)
}

func (a *A) Method(i interface{}) {
    ip := reflect.ValueOf(i)
    iv := ip.Elem()
    fmt.Println("name:", iv.Type().Name())
    for ii := 0; ii < iv.NumField(); ii++ {
        fmt.Println(iv.Field(ii))
    }
}


type B struct {
    A
    b1 int
    b2 float32
    b3 bool
}

func main() {
    var bi interface{}
    bi = &B{}
    t := reflect.TypeOf(bi).Elem()
    b := reflect.New(t)
    i, _ := b.Interface().(I)
    
    fmt.Println("----- invoke 'Method' via 'Call'. -----")
    i.Call()
    fmt.Println("----- invoke direct 'Method'. -----")
    i.Method(i)
}

/*
----- invoke 'Method' via 'Call'. -----
name: A
&lt;bool Value&gt;
----- invoke direct 'Method'. -----
name: B
&lt;main.A Value&gt;
&lt;int Value&gt;
&lt;float32 Value&gt;
&lt;bool Value&gt;
*/