Heim > Backend-Entwicklung > Golang > Wie rufe ich Methoden auf einer Schnittstelle{} in Go dynamisch auf, unabhängig vom Empfängertyp?

Wie rufe ich Methoden auf einer Schnittstelle{} in Go dynamisch auf, unabhängig vom Empfängertyp?

Barbara Streisand
Freigeben: 2024-12-02 01:39:09
Original
281 Leute haben es durchsucht

How to Dynamically Call Methods on an interface{} in Go, Regardless of Receiver Type?

Methoden auf einer Schnittstelle dynamisch aufrufen{} Unabhängig vom Empfängertyp

Im Bereich der Vorlagensysteme in Go spielt Reflexion eine entscheidende Rolle. Beim dynamischen Aufrufen von Methoden auf einer Schnittstelle{} mit unterschiedlichen Empfängertypen entsteht eine Herausforderung. Dies funktioniert zwar nahtlos mit bekannten Typen, schlägt jedoch fehl, wenn die Daten in eine Schnittstelle eingebettet sind{}.

Problembeschreibung

Bedenken Sie den folgenden Code:

type Test struct {
    Start string
}

func (t *Test) Finish() string {
    return t.Start + "finish"
}

func Pass(i interface{}) {
    _, ok := reflect.TypeOf(&i).MethodByName("Finish")
    if ok {
        fmt.Println(reflect.ValueOf(&i).MethodByName("Finish").Call([]reflect.Value{})[0])
    } else {
        fmt.Println("Pass() fail")
    }
}

func main() {
    i := Test{Start: "start"}

    Pass(i)
    _, ok := reflect.TypeOf(&i).MethodByName("Finish")
    if ok {
        fmt.Println(reflect.ValueOf(&i).MethodByName("Finish").Call([]reflect.Value{})[0])
    } else {
        fmt.Println("main() fail")
    }
}
Nach dem Login kopieren

Beobachtungen

  • Wenn es sich bei den Daten um einen bekannten Typ handelt (main()), der dynamische Methodenaufruf ist erfolgreich.
  • Wenn die Daten in eine Schnittstelle{} (Pass()) eingeschlossen werden, schlägt dies fehl und gibt beim ersten Aufruf „Pass() fail“ zurück.

Das Problem

Das Problem liegt im Zugriff auf die Adresse der Daten, wenn diese in eine Schnittstelle eingebettet sind{}. Die Verwendung von &i, das normalerweise auf einen Zeiger verweist, funktioniert in diesem Szenario nicht.

Lösung

Um dieses Problem zu beheben, müssen wir alle möglichen Szenarien behandeln:

  1. Schnittstelle{} Daten als Wert und Methodenempfänger als Wert:

    • Wenn die Daten ein Wert sind, erstellen Sie einen Zeiger darauf.
  2. Schnittstelle{} Daten als einen Wert- und Methodenempfänger als Zeiger:

    • Erstellen Sie wie oben einen Zeiger auf den Daten.
  3. Schnittstelle{} Daten als Zeiger und Methodenempfänger als Wert:

    • Wenn die Daten sind Rufen Sie als Zeiger den Wert ab, auf den er zeigt zu.
  4. Schnittstellendaten{} als Zeiger und Methodenempfänger als Zeiger:

    • Verwenden Sie den Zeiger direkt.

Implementierung

Basierend auf dieser Logik können wir eine verallgemeinerte Funktion erstellen, um Methoden dynamisch aufzurufen:

func CallMethod(i interface{}, methodName string) interface{} {
    // Handle all scenarios
    var ptr, value, finalMethod reflect.Value
    value = reflect.ValueOf(i)
    if value.Type().Kind() == reflect.Ptr {
        ptr = value
        value = ptr.Elem()
    } else {
        ptr = reflect.New(reflect.TypeOf(i))
        temp := ptr.Elem()
        temp.Set(value)
    }

    // Check for method on value and pointer
    method := value.MethodByName(methodName)
    if method.IsValid() {
        finalMethod = method
    }
    method = ptr.MethodByName(methodName)
    if method.IsValid() {
        finalMethod = method
    }

    // Call the method
    if (finalMethod.IsValid()) {
        return finalMethod.Call([]reflect.Value{})[0].Interface()
    }

    // Method not found
    return ""
}
Nach dem Login kopieren

Mit dieser Funktion können wir nun Methoden unabhängig vom Empfängertyp und der Schnittstelle dynamisch aufrufen{} Wrapper:

i := Test{Start: "start"}
j := Test{Start: "start2"}

fmt.Println(CallMethod(i, "Finish"))
// Output: startfinish
fmt.Println(CallMethod(&i, "Finish"))
// Output: startfinish
fmt.Println(CallMethod(i, "Another"))
// Output:
fmt.Println(CallMethod(&i, "Another"))
// Output: start2another
fmt.Println(CallMethod(j, "Finish"))
// Output: startfinish
fmt.Println(CallMethod(&j, "Finish"))
// Output: start2finish
fmt.Println(CallMethod(j, "Another"))
// Output:
fmt.Println(CallMethod(&j, "Another"))
// Output: start2another
Nach dem Login kopieren

Das obige ist der detaillierte Inhalt vonWie rufe ich Methoden auf einer Schnittstelle{} in Go dynamisch auf, unabhängig vom Empfängertyp?. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Quelle:php.cn
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
Neueste Artikel des Autors
Beliebte Tutorials
Mehr>
Neueste Downloads
Mehr>
Web-Effekte
Quellcode der Website
Website-Materialien
Frontend-Vorlage