D 的个人博客

但行好事莫问前程

  menu
417 文章
3446695 浏览
5 当前访客
ღゝ◡╹)ノ❤️

Go 边看边练 -《Go 学习笔记》系列(八)

上一篇: [1438260619759]


ToC


4.1 Array

和以往认知的数组有很大不同。

  • 数组是值类型,赋值和传参会复制整个数组,而不是指针。
  • 数组长度必须是常量,且是类型的组成部分。[2]int[3]int 是不同类型。
  • 支持 "=="、"!=" 操作符,因为内存总是被初始化过的。
  • 指针数组 [n]*T,数组指针 *[n]T

可用复合语句初始化。

 1a := [3]int{1, 2} // 未初始化元素值为 0。
 2b := [...]int{1, 2, 3, 4} // 通过初始化值确定数组长度。
 3c := [5]int{2: 100, 4:200} // 使用索引号初始化元素。
 4
 5d := [...]struct {
 6	name string
 7	age uint8
 8}{
 9	{"user1", 10}, // 可省略元素类型。
10	{"user2", 20}, // 别忘了最后一行的逗号。
11}

支持多维数组。

1a := [2][3]int{{1, 2, 3}, {4, 5, 6}}
2b := [...][2]int{{1, 1}, {2, 2}, {3, 3}} // 第 2 纬度不能用 "..."。

值拷贝行为会造成性能问题,通常会建议使用 slice,或数组指针。

内置函数 lencap 都返回数组长度 (元素数量)。

1a := [2]int{}
2println(len(a), cap(a)) // 2, 2

4.2 Slice

需要说明,slice 并不是数组或数组指针。它通过内部指针和相关属性引用数组片段,以实现变长方案。

runtime.h

1struct Slice
2{ // must not move anything
3	byte* array; // actual data
4	uintgo len; // number of elements
5	uintgo cap; // allocated number of elements
6};
  • 引用类型。但自身是结构体,值拷贝传递。

  • 属性 len 表示可用元素数量,读写操作不能超过该限制。

  • 属性 cap 表示最大扩张容量,不能超出数组限制。

  • 如果 slice == nil,那么 lencap 结果都等于 0

    data := [...]int{0, 1, 2, 3, 4, 5, 6}
    slice := data[1:4:5] // [low : high : max]

创建表达式使用的是元素索引号,而非数量。

读写操作实际目标是底层数组,只需注意索引号的差别。

1data := [...]int{0, 1, 2, 3, 4, 5}
2
3s := data[2:4]
4s[0] += 100
5s[1] += 200
6
7fmt.Println(s)
8fmt.Println(data)

输出:

1[102 203]
2[0 1 102 203 4 5]

可直接创建 slice 对象,自动分配底层数组。

1s1 := []int{0, 1, 2, 3, 8: 100} // 通过初始化表达式构造,可使用索引号。
2fmt.Println(s1, len(s1), cap(s1))
3
4s2 := make([]int, 6, 8) // 使用 make 创建,指定 len 和 cap 值。
5fmt.Println(s2, len(s2), cap(s2))
6
7s3 := make([]int, 6) // 省略 cap,相当于 cap = len。
8fmt.Println(s3, len(s3), cap(s3))

输出:

1[0 1 2 3 0 0 0 0 100] 9 9
2[0 0 0 0 0 0]              6 8
3[0 0 0 0 0 0]              6 6

使用 make 动态创建 slice,避免了数组必须用常量做长度的麻烦。还可用指针直接访问底层数组,退化成普通数组操作。

1s := []int{0, 1, 2, 3}
2
3p := &s[2] // *int, 获取底层数组元素指针。
4*p += 100
5
6fmt.Println(s)

输出:

1[0 1 102 3]

至于 [][]T,是指元素类型为 []T

1data := [][]int{
2	[]int{1, 2, 3},
3	[]int{100, 200},
4	[]int{11, 22, 33, 44},
5}

可直接修改 struct array/slice 成员。

 1d := [5]struct {
 2	x int
 3}{}
 4
 5s := d[:]
 6
 7d[1].x = 10
 8s[2].x = 20
 9
10fmt.Println(d)
11fmt.Printf("%p, %p\n", &d, &d[0])

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

4.2.1 reslice

所谓 reslice,是基于已有 slice 创建新 slice 对象,以便在 cap 允许范围内调整属性。

1s := []int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
2
3s1 := s[2:5] // [2 3 4]
4s2 := s1[2:6:7] // [4 5 6 7]
5s3 := s2[3:6] // Error

新对象依旧指向原底层数组。

1s := []int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
2
3s1 := s[2:5] // [2 3 4]
4s1[2] = 100
5
6s2 := s1[2:6] // [100 5 6 7]
7s2[3] = 200
8
9fmt.Println(s)

输出:

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

4.2.2 append

slice 尾部添加数据,返回新的 slice 对象。

1s := make([]int, 0, 5)
2fmt.Printf("%p\n", &s)
3
4s2 := append(s, 1)
5fmt.Printf("%p\n", &s2)
6
7fmt.Println(s, s2)

输出:

10x210230000
20x210230040
3[] [1]

简单点说,就是在 array[slice.high] 写数据。

1data := [...]int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
2
3s := data[:3]
4s2 := append(s, 100, 200) // 添加多个值。
5
6fmt.Println(data)
7fmt.Println(s)
8fmt.Println(s2)

输出:

1[0 1 2 100 200 5 6 7 8 9]
2[0 1 2]
3[0 1 2 100 200]

一旦超出原 slice.cap 限制,就会重新分配底层数组,即便原数组并未填满。

1data := [...]int{0, 1, 2, 3, 4, 10: 0}
2s := data[:2:3]
3
4s = append(s, 100, 200) // 一次 append 两个值,超出 s.cap 限制。
5
6fmt.Println(s, data) // 重新分配底层数组,与原数组无关。
7fmt.Println(&s[0], &data[0]) // 比对底层数组起始指针。

输出:

1[0 1 100 200] [0 1 2 3 4 0 0 0 0 0 0]
20x20819c180 0x20817c0c0

从输出结果可以看出,append 后的 s 重新分配了底层数组,并复制数据。如果只追加一个值,则不会超过 s.cap 限制,也就不会重新分配。

通常以 2 倍容量重新分配底层数组。在大批量添加数据时,建议一次性分配足够大的空间,以减少内存分配和数据复制开销。或初始化足够长的 len 属性,改用索引号进行操作。及时释放不再使用的 slice 对象,避免持有过期数组,造成 GC 无法回收。

 1s := make([]int, 0, 1)
 2c := cap(s)
 3
 4for i := 0; i < 50; i++ {
 5	s = append(s, i)
 6	if n := cap(s); n > c {
 7		fmt.Printf("cap: %d -> %d\n", c, n)
 8		c = n
 9	}
10}

输出:

1cap: 1 -> 2
2cap: 2 -> 4
3cap: 4 -> 8
4cap: 8 -> 16
5cap: 16 -> 32
6cap: 32 -> 64

4.2.3 copy

函数 copy 在两个 slice 间复制数据,复制长度以 len 小的为准。两个 slice 可指向同一底层数组,允许元素区间重叠。

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

下一篇: [1438596722873]



社区小贴士

  • 关注标签 [golang] 可以方便查看 Go 相关帖子
  • 关注标签 [Go 学习笔记] 可以方便查看本系列
  • 关注作者后如有新帖将会收到通知

该文章同步自 黑客派