一篇很典型的golang slice采坑记录:https://studygolang.com/articles/6557
有如下代码:
type AutoGenerated struct { Age int `json:"age"` Name string `json:"name"` Child []int `json:"child"` } func main() { jsonStr1 := "{\"age\": 12,\"name\": \"potter\", \"child\":[1,2,3]}" a := AutoGenerated{} json.Unmarshal([]byte(jsonStr1), &a) aa := a.Child fmt.Println(aa) jsonStr2 := "{\"age\": 14,\"name\": \"potter\", \"child\":[3,4,5,7,8,9]}" json.Unmarshal([]byte(jsonStr2), &a) fmt.Println(aa) }
会发现,第一次打印aa时,aa是 [1,2,3],第二次打印aa时,aa就变成了[3,4,5]
这是因为两次调用 unmarshal 时,a 里面的 Child 字段实际上是同一个 slice,刚开始第一次 unmarshal 时,a.Child = [1,2,3],其中 cap = 4。然后在第二次 unmarshal 时,给a.Child 中添加新的内容 [3,4,5,7,8,9] 时,因为 a.Child 的容量不足,因此需要重新分配底层array,但是因为 aa 对应的那个 slice 前4个位置已经被覆盖了,因此 aa 第二次打印的值就是 [1,2,3]了