使用append
先看一个例子:
1 | // 创建一个整型切片 |
输出:1
[10 20 30 60 50] [20 30 60]
下图可以非常形象的说明上述代码的运行原理:
仅做一点点小的改变,结果就不一样了:
1 | // 创建一个整型切片 |
以上代码会输出:1
[10 20 30 40 50] [0 30 60] 4
原因在于:当往 newSlice
中新增元素的时候,由于其容量不够,newSlice
会拥有一个全新的底层数组,其容量是原来的两倍(Go 会自动完成这个操作,一旦元素个数超过 1000,增长因子会设为 1.25)
使用 range
遍历 slice
在使用 range
遍历 slice
的时候,range
会创建每个元素的副本,看看这个例子:
1 | slice := []int{10, 20, 30, 40} |
输出:
1 | Value: 10 Value-Addr: C420014060 ElemAddr: C420018080 |
可以看到 Value-Addr
跟 ElemAddr
的地址是不同的,印证了上面的说法。而每次迭代的变量的地址是相同的,说明迭代过程复用了这个变量,也是一种防止内存浪费的做法。
多维切片
创建一个多维切片:1
2// 创建一个整型切片的切片
slice := [][]int{{10}, {100, 200}}
其结构可以用下图来表示:
其中第一维可以看成长度为 2,容量为 2 的保存了切片类型的切片,第二维则是整形切片。
其他规则则同处理一维切片一样了,比如:
1 | // 为第一个切片追加值为 20 的元素 |
上述操作可以用下图来表示: