go 的 `encoding/xml` 包不支持直接通过 `xml:"b.id,attr"` 这类路径语法提取子节点的属性值;需借助嵌套结构体(具名或匿名)配合 xml 标签映射来实现跨层级属性提取。
在 Go 中处理 XML 时,encoding/xml 包提供了强大且简洁的结构体标签(struct tag)机制,但其语法规则有明确限制:xml 标签不支持 XPath 式的嵌套属性路径(如 "B.id,attr")。这意味着你无法像某些其他语言(如 Python 的 lxml)那样,用单个字段标签直接绑定到深层子节点的某个属性上。
要正确提取 Something 中 B 节点的 id 属性并存入 A 结构体的某个字段(例如 Name),必须显式声明 B 的存在——即通过嵌套结构体来建模 XML 的层级关系。
type A struct {
Id string `xml:"id,attr"` // 对应
Name struct {
Id string `xml:"id,attr"` // 对应
} `xml:"B"` // 表示该匿名结构体映射到 元素
}此方式简洁,适用于 B 结构简单、无需复用的场景。解码后,a.Name.Id 即为 "B_ID"。
type A struct {
Id string `xml:"id,attr"`
Name B `xml:"B"`
}
type B struct {
Id string `xml:"id,attr"`
}这种方式更清晰、可读性更强,也便于单元测试、复用和扩展(例如后续需添加 B 的文本内容或其它属性)。
package main
import (
"encoding/xml"
"fmt"
)
const data = `Something`
type A struct {
Id string `xml:"id,attr"`
Name B
`xml:"B"`
}
type B struct {
Id string `xml:"id,attr"`
}
func main() {
var a A
if err := xml.Unmarshal([]byte(data), &a); err != nil {
panic(err)
}
fmt.Printf("A.Id: %q, B.Id: %q\n", a.Id, a.Name.Id) // 输出:A.Id: "A_ID", B.Id: "B_ID"
}总之,Go 的 XML 解析强调结构即文档模型,而非查询式抽取。合理设计嵌套结构体,是准确、健壮解析复杂 XML 的关键。