
go 的 `xml.unmarshal` 解析后,若直接用 `for _, v := range` 遍历结构体切片并赋值,修改的是副本而非原数据;需通过索引或取地址方式操作,才能使 `xml.marshal` 输出更新后的 xml。
在 Go 中处理 XML 时,一个常见误区是:成功反序列化(xml.Unmarshal)后,试图通过常规 for _, item := range slice 循环修改嵌套结构体字段(如
以下是一个修复后的完整示例,清晰展示了如何安全、准确地修改 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 内容的读取→修改→序列化全流程。










