Mendapatkan Nilai Alamat Menggunakan Refleksi
Mereflek pada antara muka membolehkan seseorang melintasi medannya. Walau bagaimanapun, mendapatkan alamat medan bukan penunjuk memberikan cabaran. Artikel ini menangani isu ini dengan mengkaji batasan penyahrujukan penunjuk dan memperkenalkan penyelesaian.
Ikhtisar Masalah
Pertimbangkan kod berikut:
<code class="go">type Z struct { Id int } type V struct { Id int F Z } type T struct { Id int F V }</code>
Fungsi InspectStruct secara rekursif melintasi struktur dan menyenaraikan butiran medannya, termasuk alamat. Berikut ialah pelaksanaan yang dipermudahkan:
<code class="go">func InspectStruct(o interface{}) { val := reflect.ValueOf(o) if val.Kind() == reflect.Interface && !val.IsNil() { val = val.Elem() } if val.Kind() == reflect.Ptr { val = val.Elem() } for i := 0; i < val.NumField(); i++ { valueField := val.Field(i) if valueField.Kind() == reflect.Ptr { valueField = valueField.Elem() } address := "not-addressable" if valueField.CanAddr() { address = fmt.Sprint(valueField.Addr().Pointer()) } fmt.Printf("Field Name: %s,\t Field Value: %v,\t Address: %v\n", val.Type().Field(i).Name, valueField.Interface(), address) if valueField.Kind() == reflect.Struct { InspectStruct(valueField.Interface()) } } }</code>
Menjalankan fungsi ini pada contoh jenis T menghasilkan:
Field Name: Id, Field Value: 1, Address: 408125440 Field Name: F, Field Value: {2 {3}}, Address: 408125444 Field Name: Id, Field Value: 2, Address: not-addressable Field Name: F, Field Value: {3}, Address: not-addressable Field Name: Id, Field Value: 3, Address: not-addressable
Seperti yang anda boleh perhatikan, alamat medan bukan penunjuk ("Id" dan "F" jenis "Z" dan "V") tidak boleh diakses.
Penyelesaian
Isu ini berpunca daripada penyahrujukan nilai yang diperoleh daripada valueField.Interface( ). Ini mengembalikan antara muka{} yang, apabila digunakan dalam nilai medan, mungkin mengakibatkan kehilangan maklumat alamat.
Berikut ialah penyelesaian yang diubah suai yang menangani perkara ini:
<code class="go">func InspectStructV(val reflect.Value) { if val.Kind() == reflect.Interface && !val.IsNil() { val = val.Elem() } if val.Kind() == reflect.Ptr { val = val.Elem() } for i := 0; i < val.NumField(); i++ { valueField := val.Field(i) if valueField.Kind() == reflect.Ptr { valueField = valueField.Elem() } address := "not-addressable" if valueField.CanAddr() { address = fmt.Sprintf("0x%X", valueField.Addr().Pointer()) } fmt.Printf("Field Name: %s,\t Field Value: %v,\t Address: %v\n", val.Type().Field(i).Name, valueField.Interface(), address) if valueField.Kind() == reflect.Struct { InspectStructV(valueField) } } } func InspectStruct(v interface{}) { InspectStructV(reflect.ValueOf(v)) }</code>
Dengan menghantar refleksi .Nilai bukannya antara muka{} untuk InspectStructV, alamat medan boleh diperoleh dengan betul.
Kesimpulan
Apabila merentasi medan bukan penunjuk dalam struktur menggunakan refleksi, adalah penting untuk mengekalkan reflect.Value dan bukannya bergantung pada valueField.Interface(). Dengan menghantar reflect.Value kepada fungsi InspectStructV rekursif, anda boleh berjaya mendapatkan semula alamat medan pada sebarang kedalaman dalam struktur.
Atas ialah kandungan terperinci Bagaimana untuk Mendapatkan Alamat Nilai Medan Bukan Penunjuk Menggunakan Refleksi dalam Go?. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!