登录
首页 >  Golang >  Go问答

根元素的 MarshalXML

来源:stackoverflow

时间:2024-04-13 13:24:34 147浏览 收藏

珍惜时间,勤奋学习!今天给大家带来《根元素的 MarshalXML》,正文内容主要涉及到等等,如果你正在学习Golang,或者是对Golang有疑问,欢迎大家关注我!后面我会持续更新相关内容的,希望都能帮到正在学习的大家!

问题内容

我似乎在让编组器为根节点调用 marshalxml 时遇到问题。我有一个用于存储 xml 的结构,如下所示:

type xmlnode struct {
    name     string            `xml:"-"`
    attrs    map[string]string `xml:"-"`
    children []xmlnode         `xml:",any"`
}

我有一个 marshalxml 函数,如下所示:

func (n *xmlnode) marshalxml(encoder *xml.encoder, start xml.startelement) error {
    fmt.println("inside marshalxml")

    // name
    start.name = xml.name{local: n.name}

    // attrs
    for k, v := range n.attrs {
        start.attr = append(start.attr, xml.attr{name: xml.name{local: k}, value: v})
    }

    type node xmlnode
    return encoder.encodeelement((*node)(n), start)
}

我用类似于以下的代码调用它:

func main() {
    root := xmlnode{name: "root", attrs: map[string]string{}}
    root.attrs["rootattr"] = "rootvalue"

    child := xmlnode{name: "child", attrs: map[string]string{}}
    child.attrs["childattr"] = "childvalue"

    root.children = append(root.children, child)
    out, _ := xml.marshal(root)
    fmt.println(string(out))
}

当它运行时,我看到以下输出:

inside marshalxml

我期待:

Inside MarshalXML
Inside MarshalXML

我知道我可以将名称编码为特定属性,但我更关心处理地图中的属性。我也可能不应该修改那里的开始元素,但它似乎适用于此。我的问题是,根元素不会调用 marshalxml,但任何子元素都会调用 marshalxml。因此,即使我在那里正确地做了一些事情,我也不确定我是否可以拦截该根元素以对其进行适当的修改。

我简单地尝试过直接使用 xml.encode ,但它似乎遵循相同的路径来构建启动模板和根,然后为 children 节点调用 marshalxml 。有没有一种干净的方法可以将 xmlnode 转换为 xml.startelement 以便我可以将其传递进去?我可以直接在 xmlnode 上编写一个函数,通过创建 startelement 并调用 encodeelementencode 来启动编组,但我试图避免这种情况并坚持使用 xml.marshal 调用,以防止其他开发人员感到困惑。 /p>

有什么想法吗?

这里有一个可运行形式的指向上述代码的 go 演示链接。


正确答案


要么通过指针,即XML.MARSHAL(&root)playground),要么将接收器类型从指针更改为非指针,即ode)元素元素playground)。

https://go.dev/ref/spec#Method_sets

T类型的方法集由声明的所有方法组成 接收器类型为 T。对应指针类型的方法集 *T 是用接收者 *TT 声明的所有方法的集合(即还包含 T 的方法集)。

类型的方法集决定了该类型可以实现的接口 实现以及可以使用该接收器调用的方法 类型。

上面的意思是 root 类型的变量 root 不实现 xml.Marshaler 因为 MarshalXML 方法不是 XmlNode 方法集的一部分。但是 &root(其类型为 *XmlNode)确实实现了 xml.Marshaler 因为 MarshalXML 方法是 *XmlNode 方法集的一部分。

当您将方法的接收器更改为非指针时,即 func (n XmlNode) MarshalXML,则类型为 rootroot 将实现 xml.Marshaler 因为 MarshalXMLzqben dczqb 方法将成为一部分XmlNode 的方法集。引用的规范还指出,&root(类型为 *XmlNode)也将实现 xml.Marshaler 因为 MarshalXML 方法即使在 XmlNode 上声明,也将成为 zqbczq 的一部分b*XmlNode 的方法设置。

请注意,当您想直接调用具体类型上的方法而不是通过接口调用时,考虑指针与非指针方法集的内容并不那么重要。例如,如果您有变量 root,其类型为 XmlNode,并且您想要调用方法 DoXyz,该方法使用指针接收器声明为 func (n *XmlNode) DoXyz(),那么您不需要 &root,的表达式 root.DoXyz 可以正常编译。

https://go.dev/ref/spec#Calls

如果x(的类型)的方法集包含m并且参数列表可以分配给m的参数列表,则方法调用x.m()是有效的。 如果 x 是可寻址的,并且 &x 的方法集包含 m,则 x.m()(&x).m() 的简写。

终于介绍完啦!小伙伴们,这篇关于《根元素的 MarshalXML》的介绍应该让你收获多多了吧!欢迎大家收藏或分享给更多需要学习的朋友吧~golang学习网公众号也会发布Golang相关知识,快来关注吧!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>