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 j := range v.Row[i].C { v.Row[i].C[j].V = "25" } }; 掌握这一内存模型细节,即可稳定、可靠地实现 XML 内容的读取→修改→序列化全流程。