golang 切片append问题

wangam · · 1874 次点击
```html 如果归根结底到长度上,那应该就解释的通了.slice的实现就是复制时,<br/> 如果不超过容量那么底层数组数据共享, 但是长度数据或者说有效位数是不共享的, 各玩各的.<br/> 如果超过容量那么底层数组数据复制一个全新的2倍容量, 使用一个全新的和原先没有任何关系的新的slice<br/> 所以就出现了上述奇怪的问题了. ```
#5
更多评论
这样的结果的确令人困惑, 能否认为这是一个golang的bug? 改成如下的代码结果会让人更加迷惑. ```go package main import "fmt" func main() { fmt.Println("Hello, World!") s1 := make([]int, 3, 10) s2 := s1 s2[0] = 100 fmt.Println(s1) fmt.Println(s2) s2 = append(s2, 1, 2, 3, 4) fmt.Println(s1, cap(s1), s2, cap(s2)) s1 = append(s1, s2...) fmt.Println(s1, cap(s1), s2, cap(s2)) s2 = append(s2, 1000) fmt.Println(s1, cap(s1), s2, cap(s2)) } 结果如下: Hello, World! [100 0 0] [100 0 0] [100 0 0] 10 [100 0 0 1 2 3 4] 10 [100 0 0 100 0 0 1 2 3 4] 10 [100 0 0 100 0 0 1] 10 [100 0 0 100 0 0 1 1000 3 4] 10 [100 0 0 100 0 0 1 1000] 10 ```
#1
slice是一段内存地址。 然后有对应有效位数和最大长度(cap) s1和s2是应对到同一段内存地址,有不同的有效位数的指。 append操作既该表 内存地址中的值,又改变了有效位数。 要避免这个问题,你需要不用使用s2:=s1这样含义不明的操作。 要复制Slice的话,使用make然后copy。
#2