

新闻资讯
行业动态Go 的 `xml.Unmarshal` 将 XML 映射为结构体后,若直接用 `for _, v := range` 遍历并赋值,实际修改的是副本而非原数据;需通过索引或取地址方式操作原始结构体字段,才能使 `xml.Marshal` 输出更新后的 XML。
在 Go 中解析并修改 XML 节点值是一个常见但易出错的操作。核心问题在于:Go 中的 range 循环默认遍历的是元素的副本(copy),而非引用。因此,即使你对循环变量(如 c.V = "25")进行了赋值,原始结构体中的对应字段并不会被修改,最终 xml.Marshal 输出的仍是初始 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"
}
}
}
// 可选:格式化输出(缩进增强可读性)
output, err := xml.MarshalIndent(&v, "", " ")
if err != nil {
fmt.Printf("marshal error: %v\n", err)
return
}
fmt.Println(string(output))
}
ow 是陷阱:r 是 v.Row[i] 的拷贝,修改 r.C[j].V 不影响 v.Row[i].C[j]。func findCell(rows []Row, targetR string) *C {
for i := range rows {
for j := range rows[i].C {
if rows[i].C[j].R == targetR {
return &rows[i].C[j]
}
}
}
return nil
}
// 使用:if c := findCell(v.Row, "B3"); c != nil { c.V = "999" }掌握「值语义 vs 引用语义」是 Go XML 处理的核心前提。只要坚持通过索引或显式取地址(&v.Row[i].C[j])操作原始数据,即可可靠地实现 XML 内容动态修改。