I'm new to golang, and am hitting some seemingly strange territory that I 
couldn't find much prior discussion about -- here's one example of the 
phenomenon in question:

https://stackoverflow.com/questions/22153269/how-to-reflect-fields-of-containing-struct-from-a-method-of-the-embedded-struct/49094937#49094937

In brief, it seems that reflect does NOT have access to the "true" 
underlying type of a struct, when accessing a function receiver struct 
pointer, but it DOES have access when an object is passed as a regular 
argument via an interface.  While I understand that the receiver is 
apparently special in terms of binding to the interface, there is also the 
statement that it is really just another arg..

In my use-case, I was trying to use reflect to provide some generic tree 
functionality that could automatically apply to "derived" types that embed 
a "Node" type with all the tree-relevant functionality.  However, depending 
on this difference in where the struct appeared (receiver vs. arg) I was 
variably getting the desired "inheritance" ability to access the fields of 
the derived types in the base type's code.  In particular, defining a 
MarshalJSON method for my base type forced everything to be Marshal'd as 
the base type, not as its actual type -- the generic Marshal gets the 
struct as an arg, whereas when you define this method you turn it into a 
receiver!

This issue generally seems relevant to this discussion about the value of 
struct embedding and the true nature of inheritance in Go: 
https://github.com/golang/go/issues/22013 -- as someone coming from the C++ 
world, I find the ability to get at least some basic inheritance 
functionality to be very powerful, while appreciating the extra flexibility 
of the compositional / interface model as well.  The example of generic 
JSON rendering shows how powerful reflection is, *when it can operate on 
the "true" underlying types* -- this peculiarity with the receivers not 
having the same "insight" into the true type seems like it should be an 
easily fixable limitation (to the extent that args really are all the 
same), and would make the overall behavior more consistent.

Philosophically, Go seems to be "optimized for the lazy" (or elegance if 
you prefer), which is certainly one of its main advantages compared to C++, 
but then in the discussion on that proposal, people seem all-too-willing to 
argue for rewriting a bunch of methods over and over again!  That seems to 
contradict this core philosophy.  Anyway, someone can hopefully let me know 
if I should file a bug ticket about this specific inconsistency (or let me 
know if it already exists somewhere -- I couldn't find it but could have 
not been using the right terms).

Here's a more succinct example based on above stackoverflow case:

https://play.golang.org/p/KmihXxU19Dd

package main

import (
"fmt"
"reflect"
)

type Inner struct {
InMbr bool
}

type InType interface {
InFun()
}

type Outer struct {
Inner
Id   int
name string
}

func (in *Inner) InFun() {
typ := reflect.TypeOf(in).Elem()
fmt.Printf("InFun on in always says Inner: %v\n", typ.Name())
// can I coerce the thing?
// it := in.(InType) // the "obvious" thing fails with: invalid type 
assertion: in.(InType) (non-interface type *Inner on left)
// try to obfuscate the process via reflect: (per 
https://groups.google.com/forum/#!msg/Golang-Nuts/KB3_Yj3Ny4c/Ai8tz-nkBwAJ)
vp := reflect.New(typ)
vp.Elem().Set(reflect.ValueOf(in).Elem())
it := vp.Interface().(InType)
ArgFun(it) // no such luck: argfun says "Inner" still
}

func ArgFun(in InType) {
typ := reflect.TypeOf(in).Elem()
fmt.Printf("ArgFun on in tells the truth: %v\n", typ.Name())
}

func main() {
val := Outer{}
val.InFun()
ArgFun(&val)
}

Output: 

InFun on in always says Inner: Inner
ArgFun on in tells the truth: Inner
ArgFun on in tells the truth: Outer


-- 
You received this message because you are subscribed to the Google Groups 
"golang-nuts" group.
To unsubscribe from this group and stop receiving emails from it, send an email 
to golang-nuts+unsubscr...@googlegroups.com.
For more options, visit https://groups.google.com/d/optout.

Reply via email to