Go 数据结构之 Slice (一)


Golang 数据结构之 Slice

前言

Slice(切片)是抽象在 Array(数组)之上的特殊的数据类型类型,在了解slice之前需要先了解一下Array的情况。

Array 数组

Array数组就是 Golang 的基本数据类型中的数字,是一种顺序存储结构。

用法

func main() {
	nums := [4]int{}
	nums[0] = 1
	nums[2] = 3

	fmt.Printf("nums: %v\n", nums)
	fmt.Printf("nums[2]: %d\n", nums[2])
}

输出结果
在这里插入图片描述

数据结构

在使用的时候需要指定长度和存储元素类型。数组在声明后,其元素的初始值(也就是零值)为 0。并且该变量可以直接使用,不需要特殊操作。底层数据存储为一段连续的内存空间,通过固定的索引值(下标)进行检索。
在这里插入图片描述
需要注意的是数组的长度是固定的,它的长度是类型的一部分,因此 [3]int 和 [4]int 在类型上是不同的

Slice 切片

Slice 是对 Array 的抽象,Slice 和 Array 不一样,它不需要指定长度。使用更加的灵活,能够自动扩容。

用法

func main() {
	nums := [4]int{}
	nums[0] = 1
	nums[2] = 3

	slice := nums[:]

	fmt.Printf("slice: %v", slice)
}

输出结果为
在这里插入图片描述

数据结构

它的存储结构这样的,共分为三部分

  • array:指向所引用的数组指针,可以说任意元素的指针
  • len:长度,当前引用切片的元素个数
  • cap:容量,当前引用切片的容量(底层数组的元素总数)
    在这里插入图片描述
    在上述代码中,可观察到 slice := nums[:],这行代码初始化了 Slice 的 Pointer 指向数组, len 和 cap 都为数组的基础属性,及长度为 4 。

len != cap

len 可以不等于 cap,但必须大于或等于 len,否则会导致 panic

func main() {
	nums := [4]int{}
	nums[0] = 1
	nums[2] = 3

	slice := nums[0:3]
	fmt.Printf("slice: %v, len: %d, cap: %d", slice, len(slice), cap(slice))
}

在这里插入图片描述数据结构是这样的,只取了 nums 的三个元素的指针
在这里插入图片描述

  • len 为所引用元素的个数;
  • cap 为所引用的数组元素总个数。

另外一种创建方式

创建切片除了上面的方式,还有另外一种不常见的方式,我们简单看一下源码

// MakeSlice creates a new zero-initialized slice value
// for the specified slice type, length, and capacity.
func MakeSlice(typ Type, len, cap int) Value {
	if typ.Kind() != Slice {
		panic("reflect.MakeSlice of non-slice type")
	}
	if len < 0 {
		panic("reflect.MakeSlice: negative len")
	}
	if cap < 0 {
		panic("reflect.MakeSlice: negative cap")
	}
	if len > cap {
		panic("reflect.MakeSlice: len > cap")
	}

	s := unsafeheader.Slice{Data: unsafe_NewArray(typ.Elem().(*rtype), cap), Len: len, Cap: cap}
	return Value{typ.(*rtype), unsafe.Pointer(&s), flagIndir | flag(Slice)}
}
  • 判断传入的 Slice 类型是否合规;
  • 判断 len ,cap 是否合规,检查 cap 是否大于 len;
  • 返回申请成功的 Slice 内存地址和相关属性。

Author: stream
Reprint policy: All articles in this blog are used except for special statements CC BY 4.0 reprint policy. If reproduced, please indicate source stream !
  TOC