Heim > Backend-Entwicklung > Golang > Wie verwende ich Reflect.NewAt auf der Schnittstelle {}?

Wie verwende ich Reflect.NewAt auf der Schnittstelle {}?

WBOY
Freigeben: 2024-02-09 14:00:24
nach vorne
956 Leute haben es durchsucht

Wie verwende ich Reflect.NewAt auf der Schnittstelle {}?

Der PHP-Editor Xinyi stellt Ihnen vor, wie Sie Reflect.NewAt auf der Schnittstelle {} verwenden. Reflect.NewAt ist eine Reflection-Bibliotheksfunktion in der Go-Sprache, die zum Erstellen einer neuen Instanz auf einem bestimmten Schnittstellentyp verwendet wird. Durch die Verwendung von „reflect.NewAt“ können wir dynamisch eine neue Schnittstelleninstanz erstellen und initialisieren, ohne zur Kompilierungszeit einen bestimmten Typ anzugeben. Dies gibt uns mehr Flexibilität und Dynamik, wodurch der Code vielseitiger und erweiterbarer wird. Im Folgenden erfahren Sie ausführlich, wie Sie mit „reflect.NewAt“ eine Instanz auf der Schnittstelle {} erstellen.

Frageninhalt

package main

import (
    "encoding/json"
    "fmt"
    "reflect"
    "unsafe"
)

type Stu struct {
    Name string `json:"name"`
}

func MakeStu() interface{} {
    return Stu{
        Name: "Test",
    }
}

func main() {
    jsonData := []byte(`{"name":"New"}`)
    t1 := MakeStu()
    t2 := MakeStu()
    t3 := MakeStu()

    json.Unmarshal(jsonData, &t1)
    fmt.Println(t1) //Type of t1 becomes map[string]interface{} instead of struct Stu

    newPointer := reflect.New(reflect.ValueOf(t2).Type()).Interface()
    json.Unmarshal(jsonData, newPointer)
    fmt.Println(newPointer) //It works,but it need allocate memory to hold temp new variable

    brokenPointer := reflect.NewAt(reflect.ValueOf(t3).Type(), unsafe.Pointer(&t3)).Interface()
    json.Unmarshal(jsonData, brokenPointer)
    fmt.Println(brokenPointer) // I want to get the pointer of original type based on the existed variable t3,but it crashes.
}
Nach dem Login kopieren

Wenn ich beim Codieren den spezifischen Typ der Schnittstelle{} nicht kenne, kann ich interface.(type) nicht zum Umwandeln verwenden. Wie verwende ich „reflect.newat“ auf der Schnittstelle {}?

Wenn es eine Schnittstelle gibt, die eine Methode enthält, die interface{} zurückgibt, ist ihr spezifischer Typ struct, aber ich kann den Typ der Struktur beim Codieren nicht bestimmen. Ich muss json.unmarshal verwenden, um von JSON codierte Daten zu dekodieren. Ich möchte nicht „map[string]interface{}“ erhalten, also muss ich den Typ des Empfängers auf einen Zeiger auf einen konkreten Typ von „interface{}“ setzen. Die Verwendung von „reflect.new“ ist einfach, verbraucht jedoch zusätzlichen Speicher. Daher bin ich neugierig, wie ich „reflect.newat“ mit einer vorhandenen Schnittstelle {} verwenden kann.

Lösung

Ich würde gerne ein paar Worte zur Erläuterung des Problems verwenden.

Funktionen, auf die wir keinen Einfluss haben makestu() 返回空接口 - 不是具体类型。这将使具体类型对 json.unmarshal() “不可见”,并且 json.unmarshal() 将其视为空接口,而不是具体类型 stu. Wir müssen dem Unmarshaller den konkreten Typ irgendwie mitteilen.

Ich würde dieses Problem mit Typschaltern lösen:

func main() {
    jsondata := []byte(`{"name":"new"}`)
    t1 := makestu()

    switch c := t1.(type) {
    default:
        _ = json.unmarshal(jsondata, &c)
        t1 = c
    }

    fmt.println(t1)
}
Nach dem Login kopieren
Der Schalter

type wandelt eine leere Schnittstelle in einen konkreten Typ um und json.unmarshal() behandelt sie als konkreten Typ. Möglicherweise verfügen Sie noch über zusätzliche Zuweisungen, aber der Code ist besser lesbar und Sie verlassen sich nicht auf Reflektion.

Wenn ich wirklich abenteuerlustig wäre, würde ich eine Hilfsfunktion hinzufügen, die einen Zeiger auf eine Nullschnittstelle akzeptiert, etwa so:

func unmarshal(data []byte, i *interface{}) (err error) {
    switch c := (*i).(type) {
    default:
        err = json.unmarshal(data, &c)
        *i = c
    }

    return err
}
Nach dem Login kopieren

Dadurch wird eine Verwendung wie folgt ermöglicht:

    jsonData := []byte(`{"name":"New"}`)
    t1 := MakeStu()
    unmarshal(jsonData, &t1)
Nach dem Login kopieren

Das sieht für mich sauber aus und erfordert keine Reflexion, aber es verwendet reflect.newat() nicht, wie Sie selbst vorgeschlagen haben. Ich hoffe, dass Sie meinen Rat weiterhin nützlich finden.

Das obige ist der detaillierte Inhalt vonWie verwende ich Reflect.NewAt auf der Schnittstelle {}?. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Verwandte Etiketten:
Quelle:stackoverflow.com
Erklärung dieser Website
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn
Beliebte Tutorials
Mehr>
Neueste Downloads
Mehr>
Web-Effekte
Quellcode der Website
Website-Materialien
Frontend-Vorlage