Dalam bidang sistem templat dalam Go, refleksi memainkan peranan yang penting. Cabaran timbul apabila memanggil kaedah secara dinamik pada antara muka{} dengan pelbagai jenis penerima. Walaupun ini berfungsi dengan lancar dengan jenis yang diketahui, ia gagal apabila data dibalut dalam antara muka{}.
Pernyataan Masalah
Pertimbangkan kod berikut:
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") } }
Pemerhatian
Masalahnya
Masalahnya terletak pada mengakses alamat data apabila ia dibungkus dalam antara muka{}. Menggunakan &i, yang biasanya menghala ke penunjuk, tidak berfungsi dalam senario ini.
Penyelesaian
Untuk menangani perkara ini, kita perlu mengendalikan semua senario yang mungkin:
antara muka{} data sebagai nilai dan penerima kaedah sebagai nilai:
antara muka{} data sebagai penerima nilai dan kaedah sebagai penunjuk:
antara muka{} data sebagai penunjuk dan penerima kaedah sebagai nilai:
antara muka{} data sebagai penunjuk dan penerima kaedah sebagai penunjuk:
Pelaksanaan
Berdasarkan logik ini, kita boleh mencipta fungsi umum untuk memanggil kaedah secara dinamik:
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 "" }
Dengan fungsi ini, kita kini boleh memanggil kaedah secara dinamik tanpa mengira jenis penerima dan antara muka{} pembalut:
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
Atas ialah kandungan terperinci Bagaimana untuk Memanggil Kaedah Secara Dinamik pada antara muka{} dalam Go, Tanpa mengira Jenis Penerima?. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!