如何在 Go 中正确修改 XML 节点的值

go 的 `xml.unmarshal` 解析后,若直接用 `for _, v := range` 遍历结构体切片并赋值,修改的是副本而非原数据;需通过索引或取地址方式操作,才能使 `xml.marshal` 输出更新后的 xml。

在 Go 中处理 XML 时,一个常见误区是:成功反序列化(xml.Unmarshal)后,试图通过常规 for _, item := range slice 循环修改嵌套结构体字段(如 对应的 C.V),却发现最终 xml.Marshal 输出的 XML 值并未改变。根本原因在于 Go 中 range 循环的每次迭代都复制元素值,而非引用原切片中的结构体实例。

以下是一个修复后的完整示例,清晰展示了如何安全、准确地修改 XML 节点值:

package main

import (
    "encoding/xml"
    "fmt"
)

type C struct {
    XMLName xml.Name `xml:"c"`
    V       string   `xml:"v,omitempty"`
    R       string   `xml:"r,attr"`
    T       string   `xml:"t,attr,omitempty"`
    S       string   `xml:"s,attr"`
}

type Row struct {
    XMLName xml.Name `xml:"row"`
    R       string   `xml:"r,attr"`
    C       []C      `xml:"c"`
    Spans   string   `xml:"spans,attr"`
}

type Result struct {
    XMLName xml.Name `xml:"sheetData"`
    Row     []Row    `xml:"row"`
}

func main() {
    input := `

  
    {{range .txt}}
    1
    2
    3
    
    21
  
  
    0
    1
    2
    3
    
    21
  
`

    var v Result
    err := xml.Unmarshal([]byte(input), &v)
    if err != nil {
        fmt.Printf("unmarshal error: %v\n", err)
        return
    }

    // ✅ 正确做法:使用索引遍历,直接修改原切片元素
    for i := range v.Row {
        for j := range v.Row[i].C {
            // 示例:将所有非空  的值统一改为 "25"
            if v.Row[i].C[j].V != "" {
                v.Row[i].C[j].V = "25"
            }
        }
    }

    // ✅ 或者更简洁:取地址后修改(语义清晰,适合复杂逻辑)
    // for i := range v.Row {
    //  for j := range v.Row[i].C {
    //      c := &v.Row[i].C[j]
    //      if c.V != "" {
    //          c.V = "25"
    //      }
    //  }
    // }

    output, err := xml.MarshalIndent(&v, "", "  ")
    if err != nil {
        fmt.Printf("marshal error: %v\n", err)
        return
    }
    fmt.Println(string(output))
}

? 关键要点总结:

  • ❌ 错误写法:for _, r := range v.Row { for _, c := range r.C { c.V = "25" } } → c 是副本,修改无效;
  • ✅ 正确写法一:for i := range v.Row { for j := range v.Row[i].C { v.Row[i].C[j].V = "25" } };
  • ✅ 正确写法二:c := &v.Row[i].C[j]; c.V = "25",利用指针明确意图,提升可读性与可维护性;
  • ⚠️ 注意:xml:"v,omitempty" 标签意味着当 V == "" 时,该 元素将被忽略(不输出),因此若需强制保留空 ,应移除 omitempty;
  • ✅ 建议始终对 xml.Unmarshal 和 xml.Marshal 的错误进行检查,避免静默失败。

掌握这一内存模型细节,即可稳定、可靠地实现 XML 内容的读取→修改→序列化全流程。