兄弟连go教程(17)数据 - Slice

23 篇文章 1 订阅

需要说明,slice 并不是数组或数组指针。它通过内部指针和相关属性引⽤用数组⽚片段,以
实现变⻓长⽅方案。
runtime.h
struct Slice
{ // must not move anything
byte* array; // actual data
uintgo len; // number of elements
uintgo cap; // allocated number of elements
};


•引⽤用类型。但⾃自⾝身是结构体,值拷⻉贝传递。
•属性 len 表⽰示可⽤用元素数量,读写操作不能超过该限制。
•属性 cap 表⽰示最⼤大扩张容量,不能超出数组限制。
•如果 slice == nil,那么 len、cap 结果都等于 0。
d
ata := [...]int{0, 1, 2, 3, 4, 5, 6}
slice := data[1:4:5] // [low : high : max]
+- low high -+ +- max len = high - low
| | | cap = max - low
+---+---+---+---+---+---+---+ +---------+---------+---------+
data | 0 | 1 | 2 | 3 | 4 | 5 | 6 | slice | pointer | len = 3 | cap = 4 |
+---+---+---+---+---+---+---+ +---------+---------+---------+
|<--- len ---->| | |
| | |
|<----- cap ------->| |
| |
+-------<<<-------- slice.array pointer ---<<<-----+



创建表达式使⽤用的是元素索引号,⽽而⾮非数量。
data := [...]int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
expression slice len cap comment
------------+----------------------+------+-------+---------------------
data[:6:8] [0 1 2 3 4 5] 6 8 省略 low.
data[5:] [5 6 7 8 9] 5 5 省略 high、max。
data[:3] [0 1 2] 3 10 省略 low、max。
data[:] [0 1 2 3 4 5 6 7 8 9] 10 10 全部省略。


读写操作实际⺫⽬目标是底层数组,只需注意索引号的差别。
data := [...]int{0, 1, 2, 3, 4, 5}
s := data[2:4]
s[0] += 100
s[1] += 200
fmt.Println(s)
fmt.Println(data)


输出:
[102 203]
[0 1 102 203 4 5]


可直接创建 slice 对象,⾃自动分配底层数组。
s1 := []int{0, 1, 2, 3, 8: 100} // 通过初始化表达式构造,可使⽤用索引号。
fmt.Println(s1, len(s1), cap(s1))
s2 := make([]int, 6, 8) // 使⽤用 make 创建,指定 len 和 cap 值。
fmt.Println(s2, len(s2), cap(s2))
s3 := make([]int, 6) // 省略 cap,相当于 cap = len。
fmt.Println(s3, len(s3), cap(s3))


输出:
[0 1 2 3 0 0 0 0 100] 9 9
[0 0 0 0 0 0] 6 8
[0 0 0 0 0 0] 6 6


使⽤用 make 动态创建 slice,避免了数组必须⽤用常量做⻓长度的⿇麻烦。还可⽤用指针直接访问
底层数组,退化成普通数组操作。
s := []int{0, 1, 2, 3}

p := &s[2] // *int, 获取底层数组元素指针。
*p += 100
fmt.Println(s)


输出:
[0 1 102 3]


至于 [][]T,是指元素类型为 []T 。
data := [][]int{
[]int{1, 2, 3},
[]int{100, 200},
[]int{11, 22, 33, 44},
}


可直接修改 struct array/slice 成员。
d := [5]struct {
x int
}{}
s := d[:]
d[1].x = 10
s[2].x = 20
fmt.Println(d)
fmt.Printf("%p, %p\n", &d, &d[0])


输出:
[{0} {10} {20} {0} {0}]
0x20819c180, 0x20819c180


1 . reslice
所谓 reslice,是基于已有 slice 创建新 slice 对象,以便在 cap 允许范围内调整属性。
s := []int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
s1 := s[2:5] // [2 3 4]
s2 := s1[2:6:7] // [4 5 6 7]
s3 := s2[3:6] // Error

+---+---+---+---+---+---+---+---+---+---+
data | 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 |
+---+---+---+---+---+---+---+---+---+---+
0 2 5
+---+---+---+---+---+---+---+---+
s1 | 2 | 3 | 4 | | | | | | len = 3, cap = 8
+---+---+---+---+---+---+---+---+
0 2 6 7
+---+---+---+---+---+
s2 | 4 | 5 | 6 | 7 | | len = 4, cap = 5
+---+---+---+---+---+
0 3 4 5
+---+---+---+
s3 | 7 | 8 | X | error: slice bounds out of range
+---+---+---+


新对象依旧指向原底层数组。
s := []int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
s1 := s[2:5] // [2 3 4]
s1[2] = 100
s2 := s1[2:6] // [100 5 6 7]
s2[3] = 200
fmt.Println(s)


输出:
[0 1 2 3 100 5 6 200 8 9]

2 .append

向 slice 尾部添加数据,
向 slice 尾部添加数据,返回新的 slice 对象。
s := make([]int, 0, 5)
fmt.Printf("%p\n", &s)
s2 := append(s, 1)
fmt.Printf("%p\n", &s2)
fmt.Println(s, s2)


输出:
0x210230000
0x210230040
[] [1]




简单点说,就是在 array[slice.high] 写数据。
data := [...]int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}

s := data[:3]
s2 := append(s, 100, 200) // 添加多个值。
fmt.Println(data)
fmt.Println(s)
fmt.Println(s2)


输出:
[0 1 2 100 200 5 6 7 8 9]
[0 1 2]
[0 1 2 100 200]


一旦超出原 slice.cap 限制,就会重新分配底层数组,即便原数组并未填满。
data := [...]int{0, 1, 2, 3, 4, 10: 0}
s := data[:2:3]
s = append(s, 100, 200) // ⼀一次 append 两个值,超出 s.cap 限制。
fmt.Println(s, data) // 重新分配底层数组,与原数组⽆无关。
fmt.Println(&s[0], &data[0]) // ⽐比对底层数组起始指针。


输出:
[0 1 100 200] [0 1 2 3 4 0 0 0 0 0 0]
0x20819c180 0x20817c0c0


从输出结果可以看出,append 后的 s 重新分配了底层数组,并复制数据。如果只追加⼀一
个值,则不会超过 s.cap 限制,也就不会重新分配。
通常以 2 倍容量重新分配底层数组。在⼤大批量添加数据时,建议⼀一次性分配⾜足够⼤大的空
间,以减少内存分配和数据复制开销。或初始化⾜足够⻓长的 len 属性,改⽤用索引号进⾏行操
作。及时释放不再使⽤用的 slice 对象,避免持有过期数组,造成 GC ⽆无法回收。
s := make([]int, 0, 1)
c := cap(s)
for i := 0; i < 50; i++ {
s = append(s, i)
if n := cap(s); n > c {
fmt.Printf("cap: %d -> %d\n", c, n)
c = n
}
}


输出:
cap: 1 -> 2
cap: 2 -> 4

cap: 4 -> 8
cap: 8 -> 16
cap: 16 -> 32
cap: 32 -> 64


3. copy
函数 copy 在两个 slice 间复制数据,复制⻓长度以 len ⼩小的为准。两个 slice 可指向同⼀一
底层数组,允许元素区间重叠。
d
ata := [...]int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
s := data[8:]
s2 := data[:5]
copy(s2, s) // dst:s2, src:s
fmt.Println(s2)
fmt.Println(data)


输出:
[8 9 2 3 4]
[8 9 2 3 4 5 6 7 8 9]


应及时将所需数据 copy 到较⼩小的 slice,以便释放超⼤大号底层数组内存。


尹成老师

QQ77025077 

微信18510341407

所有视频在尹成学院

www.yinchengxueyuan.com

尹成百度云请联系QQ475318423




评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

尹成

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值