php小編百草為你帶來了一篇關于Golang的文章,標題是“解組動態 YAML 注釋”。這篇文章將詳細介紹如何在Golang中解析包含注釋的YAML文件,并將注釋信息與對應的數據關聯起來。通過本文,你將了解到如何使用Go語言的yaml.v3包來實現這一功能,并能夠在自己的項目中靈活應用。無論你是初學者還是有一定經驗的開發者,這篇文章都將為你提供有價值的知識和技巧。讓我們一起開始吧!
問題內容
我想動態更改 struct
的注釋并使用 yaml.unmarshal
,如下所示:
package main import ( "fmt" "reflect" "gopkg.in/yaml.v3" ) type User struct { Name string `yaml:"dummy"` } func (u *User) UnmarshalYAML(node *yaml.Node) error { value := reflect.ValueOf(*u) t := value.Type() fields := make([]reflect.StructField, 0) for i := 0; i < t.NumField(); i++ { fields = append(fields, t.Field(i)) if t.Field(i).Name == "Name" { fields[i].Tag = `yaml:"name"` // Dynamic annotation } } newType := reflect.StructOf(fields) newValue := value.Convert(newType) err := node.Decode(newValue.Interface()) // Cause error because it's not pointer return err } var dat string = `name: abc` func main() { out := User{} yaml.Unmarshal([]byte(dat), &out) fmt.Printf("%+v\n", out) }
登錄后復制
它會導致像 panic:reflect:reflect.value.set using unaddressable value [recovered]
這樣的錯誤,我認為這是因為 node.decode
不與指針一起使用。那么如何創建新類型的指針呢?
解決方法
這是有效的更新演示:
package main import ( "fmt" "reflect" "unsafe" "gopkg.in/yaml.v3" ) type User struct { Name string `yaml:"dummy"` } func (u *User) UnmarshalYAML(node *yaml.Node) error { t := reflect.TypeOf(*u) fields := make([]reflect.StructField, 0) for i := 0; i < t.NumField(); i++ { fields = append(fields, t.Field(i)) if t.Field(i).Name == "Name" { fields[i].Tag = `yaml:"name"` // Dynamic annotation } } newType := reflect.StructOf(fields) newValue := reflect.NewAt(newType, unsafe.Pointer(u)).Elem() err := node.Decode(newValue.Addr().Interface()) return err } var dat string = `name: abc` func main() { out := User{} yaml.Unmarshal([]byte(dat), &out) fmt.Printf("%+v\n", out) }
登錄后復制
兩個關鍵變化:
將 newvalue.interface()
替換為 newvalue.addr().interface()
。 (參見此示例:https://www.php.cn/link/e96c7de8f6390b1e6c71556e4e0a4959 a>)
將 newvalue := value.convert(newtype)
替換為 newvalue := reflect.newat(newtype, unsafe.pointer(u)).elem()
。
我們這樣做是因為 value :=reflect.valueof(*u)
中的 value
是不可尋址的(您可以使用 fmt.printf("%v", value.addr())
進行驗證。它會出現錯誤并顯示消息 panic : 不可尋址值的reflect.value.addr(
)。