web-dev-qa-db-ja.com

Goでリフレクションを使用して構造体の名前を取得する

私はこの素晴らしい答えでこの質問を見つけました:

Golangでオブジェクトのタイプを見つける方法は?

私は答えをいじって、同じ方法で構造体の名前を取得しようとしました:

_package main

import (
        "fmt"
        "reflect"
)

type Ab struct {

}

func getType(myvar interface{}) string {
        return reflect.TypeOf(myvar).Name()
}

func main() {
        fmt.Println("Hello, playground")

        tst := "string"
        tst2 := 10
        tst3 := 1.2
        tst4 := new(Ab)

        fmt.Println(getType(tst))
        fmt.Println(getType(tst2))
        fmt.Println(getType(tst3))
        fmt.Println(getType(tst4))

}
_

遊び場に行く: http://play.golang.org/p/tD8mygvETH

しかし、出力は次のとおりです。

_Hello, playground
string
int
float64


Program exited.
_

予想される出力は次のとおりです。

_Hello, playground
string
int
float64
Ab

Program exited.
_

ドキュメントを読んで理解しようとしましたが、問題は見つかりませんでした。一般的な質問で申し訳ありませんが、

理由は何ですか、reflect.TypeOf().Name()は(this)構造体では機能しませんか?

21
Daniele D

あなたの例では、構造体型ではなく、ポインタ型(_*Ab_)の値を渡します。

Type.Name()へのこだわり

ポインタでない場合、Type.Name()は正しくAbを返します。ポインタの場合、構造体の名前が必要な場合は、Type.Elem()を使用して要素のタイプを取得できます。

_func getType(myvar interface{}) string {
    if t := reflect.TypeOf(myvar); t.Kind() == reflect.Ptr {
        return "*" + t.Elem().Name()
    } else {
        return t.Name()
    }
}
_

それをテストする:

_tst4 := Ab{}
tst5 := new(Ab)
fmt.Println(getType(tst4))
fmt.Println(getType(tst5))
_

出力(変更した例を Go Playground で試してください):

_Ab
*Ab
_

注:

Type.Name()はポインターを解決しないため、渡された値がポインターへのポインターである場合は機能しないことに注意してください。 _**Ab_は、Type.String()がポインターを自動的に解決するため、この場合も機能します。

getType()関数を_**Ab_でも(または任意の深さのポインタで)簡単に機能させることができます。

_func getType(myvar interface{}) (res string) {
    t := reflect.TypeOf(myvar)
    for t.Kind() == reflect.Ptr {
        t = t.Elem()
        res += "*"
    }
    return res + t.Name()
}
_

値で呼び出す:

_tst4 := Ab{}
tst5 := new(Ab)
tst6 := &tst5 // type of **Ab
tst7 := &tst6 // type of ***Ab
_

出力( Go Playground で試してください):

_Ab
*Ab
**Ab
***Ab
_

Type.String()の使用

より簡単で優れたアプローチは、Type.String()の代わりにType.Name()を使用することです。これは、ポインターを自動的に処理し、パッケージ名も含みます。例えば。:

_func getType(myvar interface{}) string {
    return reflect.TypeOf(myvar).String()
}
_

変更された例では、次のように出力されます。

_string
int
float64
main.Ab
*main.Ab
_

Go Playground でこのバリアントを試してください。

44
icza

fmtにはクールな%Tタグも

package main

import (
    "fmt"
    "net/http"
)

type Potato struct {
}

func main() {
    fmt.Printf("I have a %T, an %T and a %T\n", Potato{}, http.StatusMultipleChoices, &http.Response{})
}

出力I have a main.Potato, an int and a *http.Responsehttps://play.golang.org/p/6z7_0BSitm

6
Azr

問題はnewがポインターを返すことです。

package main

import (
    "fmt"
    "reflect"
)

type Ab struct {
}

func getType(myvar interface{}) {
    valueOf := reflect.ValueOf(myvar)

    if valueOf.Type().Kind() == reflect.Ptr {
        fmt.Println(reflect.Indirect(valueOf).Type().Name())
    } else {
        fmt.Println(valueOf.Type().Name())
    }
}

func main() {
    fmt.Println("Hello, playground")

    tst := "string"
    tst2 := 10
    tst3 := 1.2
    tst4 := new(Ab)

    getType(tst)
    getType(tst2)
    getType(tst3)
    getType(tst4)

}

出力は

Hello, playground
string
int
float64
Ab
2
sfault